@monadns/sdk 2.3.2 → 2.3.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +1 -1
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/react.cjs +1 -1
- package/dist/react.cjs.map +1 -1
- package/dist/react.js +1 -1
- package/dist/react.js.map +1 -1
- package/package.json +1 -1
package/dist/index.cjs
CHANGED
|
@@ -595,7 +595,7 @@ async function getRegisterTx(label, ownerAddress, durationYears = 1, config) {
|
|
|
595
595
|
functionName: "register",
|
|
596
596
|
args: [label, ownerAddress, info.duration],
|
|
597
597
|
value: info.isFreebie ? 0n : info.price,
|
|
598
|
-
gas:
|
|
598
|
+
gas: 800000n,
|
|
599
599
|
_meta: {
|
|
600
600
|
label,
|
|
601
601
|
isFreebie: info.isFreebie,
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["// Core resolution (read-only)\nexport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getDisplayName,\n resolveInput,\n getAvailable,\n getOwner,\n getResolver,\n getExpiry,\n clearCache,\n} from './resolve';\n\n// Write helpers\nexport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n controllerReadAbi,\n controllerWriteAbi,\n resolverWriteAbi,\n} from './write';\nexport type { RegistrationInfo } from './write';\n\n// Client config\nexport { getMNSClient } from './client';\nexport type { MNSClientConfig } from './client';\n\n// Constants\nexport {\n MNS_REGISTRY,\n MNS_PUBLIC_RESOLVER,\n MNS_CONTROLLER,\n MNS_BASE_REGISTRAR,\n MAX_AVATAR_BYTES,\n DEFAULT_AVATAR_PLACEHOLDER,\n} from './constants';\n\n// Utilities\nexport {\n validateAvatarUri,\n validateAvatarFull,\n getRemoteAvatarSize,\n validateMonName,\n} from './utils';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 3_000_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["import_viem","import_viem"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["// Core resolution (read-only)\nexport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getDisplayName,\n resolveInput,\n getAvailable,\n getOwner,\n getResolver,\n getExpiry,\n clearCache,\n} from './resolve';\n\n// Write helpers\nexport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n controllerReadAbi,\n controllerWriteAbi,\n resolverWriteAbi,\n} from './write';\nexport type { RegistrationInfo } from './write';\n\n// Client config\nexport { getMNSClient } from './client';\nexport type { MNSClientConfig } from './client';\n\n// Constants\nexport {\n MNS_REGISTRY,\n MNS_PUBLIC_RESOLVER,\n MNS_CONTROLLER,\n MNS_BASE_REGISTRAR,\n MAX_AVATAR_BYTES,\n DEFAULT_AVATAR_PLACEHOLDER,\n} from './constants';\n\n// Utilities\nexport {\n validateAvatarUri,\n validateAvatarFull,\n getRemoteAvatarSize,\n validateMonName,\n} from './utils';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 800_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["import_viem","import_viem"]}
|
package/dist/index.js
CHANGED
|
@@ -543,7 +543,7 @@ async function getRegisterTx(label, ownerAddress, durationYears = 1, config) {
|
|
|
543
543
|
functionName: "register",
|
|
544
544
|
args: [label, ownerAddress, info.duration],
|
|
545
545
|
value: info.isFreebie ? 0n : info.price,
|
|
546
|
-
gas:
|
|
546
|
+
gas: 800000n,
|
|
547
547
|
_meta: {
|
|
548
548
|
label,
|
|
549
549
|
isFreebie: info.isFreebie,
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 3_000_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["namehash","namehash"]}
|
|
1
|
+
{"version":3,"sources":["../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 800_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["namehash","namehash"]}
|
package/dist/react.cjs
CHANGED
|
@@ -554,7 +554,7 @@ async function getRegisterTx(label, ownerAddress, durationYears = 1, config) {
|
|
|
554
554
|
functionName: "register",
|
|
555
555
|
args: [label, ownerAddress, info.duration],
|
|
556
556
|
value: info.isFreebie ? 0n : info.price,
|
|
557
|
-
gas:
|
|
557
|
+
gas: 800000n,
|
|
558
558
|
_meta: {
|
|
559
559
|
label,
|
|
560
560
|
isFreebie: info.isFreebie,
|
package/dist/react.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/react-write.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getOwner,\n getResolver,\n getExpiry,\n} from './resolve';\nimport type { MNSClientConfig } from './client';\n\nexport type { MNSClientConfig };\n\n/**\n * Reverse resolution hook: address → .mon name\n *\n * @example\n * ```tsx\n * function Profile({ address }: { address: string }) {\n * const { name, loading } = useMNSName(address)\n * return <span>{loading ? '...' : name ?? address}</span>\n * }\n * ```\n */\nexport function useMNSName(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!address || !address.startsWith('0x') || address.length !== 42) {\n setName(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n lookupAddress(address, config)\n .then((n) => {\n if (!cancelled) setName(n);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [address]);\n\n return { name, loading, error };\n}\n\n/**\n * Forward resolution hook: .mon name → address\n *\n * @example\n * ```tsx\n * function Lookup() {\n * const { address, loading } = useMNSAddress('alice.mon')\n * return <span>{loading ? 'Resolving...' : address}</span>\n * }\n * ```\n */\nexport function useMNSAddress(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [address, setAddress] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setAddress(null);\n return;\n }\n const fullName = name.endsWith('.mon') ? name : `${name}.mon`;\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n resolveName(fullName, config)\n .then((a) => {\n if (!cancelled) setAddress(a);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { address, loading, error };\n}\n\n/**\n * Display name hook: .mon name if available, otherwise truncated address\n *\n * @example\n * ```tsx\n * function Badge({ address }: { address: string }) {\n * const { displayName, monName } = useMNSDisplay(address)\n * return (\n * <span className={monName ? 'text-purple-400' : 'text-gray-500'}>\n * {displayName}\n * </span>\n * )\n * }\n * ```\n */\nexport function useMNSDisplay(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const { name, loading, error } = useMNSName(address, config);\n const truncated = address\n ? `${address.slice(0, 6)}...${address.slice(-4)}`\n : '';\n\n return {\n displayName: name || truncated,\n monName: name,\n loading,\n error,\n };\n}\n\n/**\n * On-demand resolver hook: for input fields accepting names or addresses\n *\n * @example\n * ```tsx\n * function SendForm() {\n * const { resolve, address, name, loading, error } = useMNSResolve()\n * return (\n * <div>\n * <input placeholder=\"Address or .mon name\" onChange={(e) => resolve(e.target.value)} />\n * {loading && <span>Resolving...</span>}\n * {address && <span>→ {address}</span>}\n * </div>\n * )\n * }\n * ```\n */\nexport function useMNSResolve(config?: MNSClientConfig) {\n const [address, setAddress] = useState<string | null>(null);\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const resolve = useCallback(async (input: string) => {\n const trimmed = input.trim();\n setError(null);\n\n if (!trimmed) {\n setAddress(null);\n setName(null);\n return;\n }\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) {\n setAddress(trimmed);\n setLoading(true);\n const n = await lookupAddress(trimmed, config);\n setName(n);\n setLoading(false);\n return;\n }\n\n const fullName = trimmed.endsWith('.mon') ? trimmed : `${trimmed}.mon`;\n setName(fullName);\n setLoading(true);\n\n try {\n const addr = await resolveName(fullName, config);\n setAddress(addr);\n if (!addr) setError(`${fullName} not found`);\n } catch (e: any) {\n setError(e.message);\n setAddress(null);\n } finally {\n setLoading(false);\n }\n }, []);\n\n return { resolve, address, name, loading, error };\n}\n\n/**\n * Text record hook: get metadata like avatar, URL, twitter handle\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { value } = useMNSText(name, 'avatar')\n * return value ? <img src={value} /> : null\n * }\n * ```\n */\nexport function useMNSText(\n name: string | undefined,\n key: string,\n config?: MNSClientConfig,\n) {\n const [value, setValue] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setValue(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getTextRecord(name, key, config)\n .then((v) => {\n if (!cancelled) setValue(v);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name, key]);\n\n return { value, loading, error };\n}\n\n/**\n * Avatar URL hook: resolves avatar from text record, handling IPFS and NFT formats\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { url, loading } = useMNSAvatar(name)\n * if (loading) return <div className=\"animate-pulse w-10 h-10 rounded-full bg-gray-700\" />\n * return url ? <img src={url} className=\"w-10 h-10 rounded-full\" /> : null\n * }\n * ```\n */\nexport function useMNSAvatar(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [url, setUrl] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setUrl(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getAvatarUrl(name, config)\n .then((u) => {\n if (!cancelled) setUrl(u);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { url, loading, error };\n}\n\n/**\n * Owner hook: get the owner address of a .mon name\n *\n * @example\n * ```tsx\n * function OwnerInfo({ name }: { name: string }) {\n * const { owner, loading } = useMNSOwner(name)\n * if (loading) return <div>Loading...</div>\n * return <div>Owner: {owner}</div>\n * }\n * ```\n */\nexport function useMNSOwner(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [owner, setOwner] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setOwner(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getOwner(name, config)\n .then((o) => {\n if (!cancelled) setOwner(o);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { owner, loading, error };\n}\n\n/**\n * Resolver hook: get the resolver contract address for a .mon name\n *\n * @example\n * ```tsx\n * function ResolverInfo({ name }: { name: string }) {\n * const { resolver, loading } = useMNSResolver(name)\n * return <div>Resolver: {resolver}</div>\n * }\n * ```\n */\nexport function useMNSResolver(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [resolver, setResolver] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setResolver(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getResolver(name, config)\n .then((r) => {\n if (!cancelled) setResolver(r);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { resolver, loading, error };\n}\n\n/**\n * Expiry hook: get expiration info for a .mon name\n *\n * @example\n * ```tsx\n * function ExpiryInfo({ name }: { name: string }) {\n * const { expiry, expiryDate, daysUntilExpiry, isExpired, loading } = useMNSExpiry(name)\n * if (loading) return <div>Loading...</div>\n * if (isExpired) return <div className=\"text-red-500\">Expired!</div>\n * return <div>Expires in {daysUntilExpiry} days</div>\n * }\n * ```\n */\nexport function useMNSExpiry(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [expiry, setExpiry] = useState<number | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setExpiry(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getExpiry(name, config)\n .then((e) => {\n if (!cancelled) setExpiry(e);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n const expiryDate = expiry ? new Date(expiry * 1000) : null;\n const daysUntilExpiry = expiry\n ? Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n : null;\n const isExpired = expiry ? expiry * 1000 < Date.now() : null;\n\n return { expiry, expiryDate, daysUntilExpiry, isExpired, loading, error };\n}\n\n// Write hooks\nexport {\n useRegistrationInfo,\n useMNSRegister,\n useMNSTextRecords,\n useMNSAddr,\n} from './react-write';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n type RegistrationInfo,\n} from './write';\nimport type { MNSClientConfig } from './client';\nimport { validateAvatarFull, validateMonName } from './utils';\nimport { MAX_AVATAR_BYTES } from './constants';\n\nexport { TEXT_RECORD_KEYS, MAX_AVATAR_BYTES };\nexport type { RegistrationInfo };\n\n/**\n * Hook to get registration info (price, availability, freebie status).\n * Use this to build a registration UI before the user submits.\n *\n * @example\n * ```tsx\n * function RegisterPage() {\n * const [name, setName] = useState('')\n * const { address } = useAccount()\n * const { info, loading, error } = useRegistrationInfo(name, address)\n *\n * return (\n * <div>\n * <input value={name} onChange={(e) => setName(e.target.value)} />\n * {info?.available && (\n * <p>{info.isFreebie ? 'FREE!' : `${formatEther(info.price)} MON`}</p>\n * )}\n * </div>\n * )\n * }\n * ```\n */\nexport function useRegistrationInfo(\n label: string | undefined,\n userAddress: string | undefined,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n const [info, setInfo] = useState<RegistrationInfo | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!label || !userAddress || label.trim().length < 3) {\n setInfo(null);\n setError(null);\n setLoading(false);\n return;\n }\n\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n setInfo(null);\n setError(validation.error || 'Invalid name');\n setLoading(false);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n const timer = setTimeout(() => {\n getRegistrationInfo(label, userAddress, durationYears, config)\n .then((i) => {\n if (!cancelled) setInfo(i);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n }, 300); // debounce\n\n return () => {\n cancelled = true;\n clearTimeout(timer);\n };\n }, [label, userAddress, durationYears, config]);\n\n return { info, loading, error };\n}\n\n// Type for the registration transaction\ntype RegisterTx = Awaited<ReturnType<typeof getRegisterTx>>;\n\n/**\n * Hook to register a .mon name.\n * Returns a prepare function that gives you tx params for wagmi's writeContract.\n *\n * @example\n * ```tsx\n * const { prepare, loading, error } = useMNSRegister()\n * const { writeContract, data: hash } = useWriteContract()\n *\n * const handleRegister = async () => {\n * const tx = await prepare('alice', address)\n * if (tx) writeContract(tx)\n * }\n * ```\n */\nexport function useMNSRegister(config?: MNSClientConfig) {\n const [tx, setTx] = useState<RegisterTx | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const prepare = useCallback(\n async (label: string, ownerAddress: string, durationYears = 1) => {\n setLoading(true);\n setError(null);\n setTx(null);\n\n try {\n const txParams = await getRegisterTx(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n setTx(txParams);\n return txParams;\n } catch (e: any) {\n setError(e.message);\n return null;\n } finally {\n setLoading(false);\n }\n },\n [],\n );\n\n return { prepare, tx, loading, error };\n}\n\n/**\n * Hook to set text records (avatar, url, twitter, etc.)\n *\n * @example\n * ```tsx\n * const { setAvatar, setTwitter } = useMNSTextRecords()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAvatar('alice.mon', 'https://example.com/pic.png'))\n * ```\n */\nexport function useMNSTextRecords() {\n const setTextRecord = useCallback(\n (name: string, key: string, value: string) => {\n return getSetTextTx(name, key, value);\n },\n [],\n );\n\n const setAvatar = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setAvatarValidated = useCallback(async (name: string, url: string) => {\n await validateAvatarFull(url); // Throws if invalid\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setUrl = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'url', url);\n }, []);\n\n const setTwitter = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.twitter', handle);\n }, []);\n\n const setGithub = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.github', handle);\n }, []);\n\n const setDiscord = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.discord', handle);\n }, []);\n\n const setDescription = useCallback((name: string, desc: string) => {\n return getSetTextTx(name, 'description', desc);\n }, []);\n\n return {\n setTextRecord,\n setAvatar,\n setAvatarValidated,\n setUrl,\n setTwitter,\n setGithub,\n setDiscord,\n setDescription,\n textRecordKeys: TEXT_RECORD_KEYS,\n };\n}\n\n/**\n * Hook to update the address a name points to.\n *\n * @example\n * ```tsx\n * const { setAddr } = useMNSAddr()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAddr('alice.mon', '0xnewaddress...'))\n * ```\n */\nexport function useMNSAddr() {\n const setAddr = useCallback((name: string, addr: string) => {\n return getSetAddrTx(name, addr);\n }, []);\n\n return { setAddr };\n}\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 3_000_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["import_react","import_viem","import_viem"]}
|
|
1
|
+
{"version":3,"sources":["../src/react.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/react-write.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getOwner,\n getResolver,\n getExpiry,\n} from './resolve';\nimport type { MNSClientConfig } from './client';\n\nexport type { MNSClientConfig };\n\n/**\n * Reverse resolution hook: address → .mon name\n *\n * @example\n * ```tsx\n * function Profile({ address }: { address: string }) {\n * const { name, loading } = useMNSName(address)\n * return <span>{loading ? '...' : name ?? address}</span>\n * }\n * ```\n */\nexport function useMNSName(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!address || !address.startsWith('0x') || address.length !== 42) {\n setName(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n lookupAddress(address, config)\n .then((n) => {\n if (!cancelled) setName(n);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [address]);\n\n return { name, loading, error };\n}\n\n/**\n * Forward resolution hook: .mon name → address\n *\n * @example\n * ```tsx\n * function Lookup() {\n * const { address, loading } = useMNSAddress('alice.mon')\n * return <span>{loading ? 'Resolving...' : address}</span>\n * }\n * ```\n */\nexport function useMNSAddress(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [address, setAddress] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setAddress(null);\n return;\n }\n const fullName = name.endsWith('.mon') ? name : `${name}.mon`;\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n resolveName(fullName, config)\n .then((a) => {\n if (!cancelled) setAddress(a);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { address, loading, error };\n}\n\n/**\n * Display name hook: .mon name if available, otherwise truncated address\n *\n * @example\n * ```tsx\n * function Badge({ address }: { address: string }) {\n * const { displayName, monName } = useMNSDisplay(address)\n * return (\n * <span className={monName ? 'text-purple-400' : 'text-gray-500'}>\n * {displayName}\n * </span>\n * )\n * }\n * ```\n */\nexport function useMNSDisplay(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const { name, loading, error } = useMNSName(address, config);\n const truncated = address\n ? `${address.slice(0, 6)}...${address.slice(-4)}`\n : '';\n\n return {\n displayName: name || truncated,\n monName: name,\n loading,\n error,\n };\n}\n\n/**\n * On-demand resolver hook: for input fields accepting names or addresses\n *\n * @example\n * ```tsx\n * function SendForm() {\n * const { resolve, address, name, loading, error } = useMNSResolve()\n * return (\n * <div>\n * <input placeholder=\"Address or .mon name\" onChange={(e) => resolve(e.target.value)} />\n * {loading && <span>Resolving...</span>}\n * {address && <span>→ {address}</span>}\n * </div>\n * )\n * }\n * ```\n */\nexport function useMNSResolve(config?: MNSClientConfig) {\n const [address, setAddress] = useState<string | null>(null);\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const resolve = useCallback(async (input: string) => {\n const trimmed = input.trim();\n setError(null);\n\n if (!trimmed) {\n setAddress(null);\n setName(null);\n return;\n }\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) {\n setAddress(trimmed);\n setLoading(true);\n const n = await lookupAddress(trimmed, config);\n setName(n);\n setLoading(false);\n return;\n }\n\n const fullName = trimmed.endsWith('.mon') ? trimmed : `${trimmed}.mon`;\n setName(fullName);\n setLoading(true);\n\n try {\n const addr = await resolveName(fullName, config);\n setAddress(addr);\n if (!addr) setError(`${fullName} not found`);\n } catch (e: any) {\n setError(e.message);\n setAddress(null);\n } finally {\n setLoading(false);\n }\n }, []);\n\n return { resolve, address, name, loading, error };\n}\n\n/**\n * Text record hook: get metadata like avatar, URL, twitter handle\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { value } = useMNSText(name, 'avatar')\n * return value ? <img src={value} /> : null\n * }\n * ```\n */\nexport function useMNSText(\n name: string | undefined,\n key: string,\n config?: MNSClientConfig,\n) {\n const [value, setValue] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setValue(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getTextRecord(name, key, config)\n .then((v) => {\n if (!cancelled) setValue(v);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name, key]);\n\n return { value, loading, error };\n}\n\n/**\n * Avatar URL hook: resolves avatar from text record, handling IPFS and NFT formats\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { url, loading } = useMNSAvatar(name)\n * if (loading) return <div className=\"animate-pulse w-10 h-10 rounded-full bg-gray-700\" />\n * return url ? <img src={url} className=\"w-10 h-10 rounded-full\" /> : null\n * }\n * ```\n */\nexport function useMNSAvatar(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [url, setUrl] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setUrl(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getAvatarUrl(name, config)\n .then((u) => {\n if (!cancelled) setUrl(u);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { url, loading, error };\n}\n\n/**\n * Owner hook: get the owner address of a .mon name\n *\n * @example\n * ```tsx\n * function OwnerInfo({ name }: { name: string }) {\n * const { owner, loading } = useMNSOwner(name)\n * if (loading) return <div>Loading...</div>\n * return <div>Owner: {owner}</div>\n * }\n * ```\n */\nexport function useMNSOwner(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [owner, setOwner] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setOwner(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getOwner(name, config)\n .then((o) => {\n if (!cancelled) setOwner(o);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { owner, loading, error };\n}\n\n/**\n * Resolver hook: get the resolver contract address for a .mon name\n *\n * @example\n * ```tsx\n * function ResolverInfo({ name }: { name: string }) {\n * const { resolver, loading } = useMNSResolver(name)\n * return <div>Resolver: {resolver}</div>\n * }\n * ```\n */\nexport function useMNSResolver(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [resolver, setResolver] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setResolver(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getResolver(name, config)\n .then((r) => {\n if (!cancelled) setResolver(r);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { resolver, loading, error };\n}\n\n/**\n * Expiry hook: get expiration info for a .mon name\n *\n * @example\n * ```tsx\n * function ExpiryInfo({ name }: { name: string }) {\n * const { expiry, expiryDate, daysUntilExpiry, isExpired, loading } = useMNSExpiry(name)\n * if (loading) return <div>Loading...</div>\n * if (isExpired) return <div className=\"text-red-500\">Expired!</div>\n * return <div>Expires in {daysUntilExpiry} days</div>\n * }\n * ```\n */\nexport function useMNSExpiry(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [expiry, setExpiry] = useState<number | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setExpiry(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getExpiry(name, config)\n .then((e) => {\n if (!cancelled) setExpiry(e);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n const expiryDate = expiry ? new Date(expiry * 1000) : null;\n const daysUntilExpiry = expiry\n ? Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n : null;\n const isExpired = expiry ? expiry * 1000 < Date.now() : null;\n\n return { expiry, expiryDate, daysUntilExpiry, isExpired, loading, error };\n}\n\n// Write hooks\nexport {\n useRegistrationInfo,\n useMNSRegister,\n useMNSTextRecords,\n useMNSAddr,\n} from './react-write';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n type RegistrationInfo,\n} from './write';\nimport type { MNSClientConfig } from './client';\nimport { validateAvatarFull, validateMonName } from './utils';\nimport { MAX_AVATAR_BYTES } from './constants';\n\nexport { TEXT_RECORD_KEYS, MAX_AVATAR_BYTES };\nexport type { RegistrationInfo };\n\n/**\n * Hook to get registration info (price, availability, freebie status).\n * Use this to build a registration UI before the user submits.\n *\n * @example\n * ```tsx\n * function RegisterPage() {\n * const [name, setName] = useState('')\n * const { address } = useAccount()\n * const { info, loading, error } = useRegistrationInfo(name, address)\n *\n * return (\n * <div>\n * <input value={name} onChange={(e) => setName(e.target.value)} />\n * {info?.available && (\n * <p>{info.isFreebie ? 'FREE!' : `${formatEther(info.price)} MON`}</p>\n * )}\n * </div>\n * )\n * }\n * ```\n */\nexport function useRegistrationInfo(\n label: string | undefined,\n userAddress: string | undefined,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n const [info, setInfo] = useState<RegistrationInfo | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!label || !userAddress || label.trim().length < 3) {\n setInfo(null);\n setError(null);\n setLoading(false);\n return;\n }\n\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n setInfo(null);\n setError(validation.error || 'Invalid name');\n setLoading(false);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n const timer = setTimeout(() => {\n getRegistrationInfo(label, userAddress, durationYears, config)\n .then((i) => {\n if (!cancelled) setInfo(i);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n }, 300); // debounce\n\n return () => {\n cancelled = true;\n clearTimeout(timer);\n };\n }, [label, userAddress, durationYears, config]);\n\n return { info, loading, error };\n}\n\n// Type for the registration transaction\ntype RegisterTx = Awaited<ReturnType<typeof getRegisterTx>>;\n\n/**\n * Hook to register a .mon name.\n * Returns a prepare function that gives you tx params for wagmi's writeContract.\n *\n * @example\n * ```tsx\n * const { prepare, loading, error } = useMNSRegister()\n * const { writeContract, data: hash } = useWriteContract()\n *\n * const handleRegister = async () => {\n * const tx = await prepare('alice', address)\n * if (tx) writeContract(tx)\n * }\n * ```\n */\nexport function useMNSRegister(config?: MNSClientConfig) {\n const [tx, setTx] = useState<RegisterTx | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const prepare = useCallback(\n async (label: string, ownerAddress: string, durationYears = 1) => {\n setLoading(true);\n setError(null);\n setTx(null);\n\n try {\n const txParams = await getRegisterTx(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n setTx(txParams);\n return txParams;\n } catch (e: any) {\n setError(e.message);\n return null;\n } finally {\n setLoading(false);\n }\n },\n [],\n );\n\n return { prepare, tx, loading, error };\n}\n\n/**\n * Hook to set text records (avatar, url, twitter, etc.)\n *\n * @example\n * ```tsx\n * const { setAvatar, setTwitter } = useMNSTextRecords()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAvatar('alice.mon', 'https://example.com/pic.png'))\n * ```\n */\nexport function useMNSTextRecords() {\n const setTextRecord = useCallback(\n (name: string, key: string, value: string) => {\n return getSetTextTx(name, key, value);\n },\n [],\n );\n\n const setAvatar = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setAvatarValidated = useCallback(async (name: string, url: string) => {\n await validateAvatarFull(url); // Throws if invalid\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setUrl = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'url', url);\n }, []);\n\n const setTwitter = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.twitter', handle);\n }, []);\n\n const setGithub = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.github', handle);\n }, []);\n\n const setDiscord = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.discord', handle);\n }, []);\n\n const setDescription = useCallback((name: string, desc: string) => {\n return getSetTextTx(name, 'description', desc);\n }, []);\n\n return {\n setTextRecord,\n setAvatar,\n setAvatarValidated,\n setUrl,\n setTwitter,\n setGithub,\n setDiscord,\n setDescription,\n textRecordKeys: TEXT_RECORD_KEYS,\n };\n}\n\n/**\n * Hook to update the address a name points to.\n *\n * @example\n * ```tsx\n * const { setAddr } = useMNSAddr()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAddr('alice.mon', '0xnewaddress...'))\n * ```\n */\nexport function useMNSAddr() {\n const setAddr = useCallback((name: string, addr: string) => {\n return getSetAddrTx(name, addr);\n }, []);\n\n return { setAddr };\n}\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 800_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["import_react","import_viem","import_viem"]}
|
package/dist/react.js
CHANGED
|
@@ -522,7 +522,7 @@ async function getRegisterTx(label, ownerAddress, durationYears = 1, config) {
|
|
|
522
522
|
functionName: "register",
|
|
523
523
|
args: [label, ownerAddress, info.duration],
|
|
524
524
|
value: info.isFreebie ? 0n : info.price,
|
|
525
|
-
gas:
|
|
525
|
+
gas: 800000n,
|
|
526
526
|
_meta: {
|
|
527
527
|
label,
|
|
528
528
|
isFreebie: info.isFreebie,
|
package/dist/react.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/react-write.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getOwner,\n getResolver,\n getExpiry,\n} from './resolve';\nimport type { MNSClientConfig } from './client';\n\nexport type { MNSClientConfig };\n\n/**\n * Reverse resolution hook: address → .mon name\n *\n * @example\n * ```tsx\n * function Profile({ address }: { address: string }) {\n * const { name, loading } = useMNSName(address)\n * return <span>{loading ? '...' : name ?? address}</span>\n * }\n * ```\n */\nexport function useMNSName(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!address || !address.startsWith('0x') || address.length !== 42) {\n setName(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n lookupAddress(address, config)\n .then((n) => {\n if (!cancelled) setName(n);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [address]);\n\n return { name, loading, error };\n}\n\n/**\n * Forward resolution hook: .mon name → address\n *\n * @example\n * ```tsx\n * function Lookup() {\n * const { address, loading } = useMNSAddress('alice.mon')\n * return <span>{loading ? 'Resolving...' : address}</span>\n * }\n * ```\n */\nexport function useMNSAddress(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [address, setAddress] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setAddress(null);\n return;\n }\n const fullName = name.endsWith('.mon') ? name : `${name}.mon`;\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n resolveName(fullName, config)\n .then((a) => {\n if (!cancelled) setAddress(a);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { address, loading, error };\n}\n\n/**\n * Display name hook: .mon name if available, otherwise truncated address\n *\n * @example\n * ```tsx\n * function Badge({ address }: { address: string }) {\n * const { displayName, monName } = useMNSDisplay(address)\n * return (\n * <span className={monName ? 'text-purple-400' : 'text-gray-500'}>\n * {displayName}\n * </span>\n * )\n * }\n * ```\n */\nexport function useMNSDisplay(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const { name, loading, error } = useMNSName(address, config);\n const truncated = address\n ? `${address.slice(0, 6)}...${address.slice(-4)}`\n : '';\n\n return {\n displayName: name || truncated,\n monName: name,\n loading,\n error,\n };\n}\n\n/**\n * On-demand resolver hook: for input fields accepting names or addresses\n *\n * @example\n * ```tsx\n * function SendForm() {\n * const { resolve, address, name, loading, error } = useMNSResolve()\n * return (\n * <div>\n * <input placeholder=\"Address or .mon name\" onChange={(e) => resolve(e.target.value)} />\n * {loading && <span>Resolving...</span>}\n * {address && <span>→ {address}</span>}\n * </div>\n * )\n * }\n * ```\n */\nexport function useMNSResolve(config?: MNSClientConfig) {\n const [address, setAddress] = useState<string | null>(null);\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const resolve = useCallback(async (input: string) => {\n const trimmed = input.trim();\n setError(null);\n\n if (!trimmed) {\n setAddress(null);\n setName(null);\n return;\n }\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) {\n setAddress(trimmed);\n setLoading(true);\n const n = await lookupAddress(trimmed, config);\n setName(n);\n setLoading(false);\n return;\n }\n\n const fullName = trimmed.endsWith('.mon') ? trimmed : `${trimmed}.mon`;\n setName(fullName);\n setLoading(true);\n\n try {\n const addr = await resolveName(fullName, config);\n setAddress(addr);\n if (!addr) setError(`${fullName} not found`);\n } catch (e: any) {\n setError(e.message);\n setAddress(null);\n } finally {\n setLoading(false);\n }\n }, []);\n\n return { resolve, address, name, loading, error };\n}\n\n/**\n * Text record hook: get metadata like avatar, URL, twitter handle\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { value } = useMNSText(name, 'avatar')\n * return value ? <img src={value} /> : null\n * }\n * ```\n */\nexport function useMNSText(\n name: string | undefined,\n key: string,\n config?: MNSClientConfig,\n) {\n const [value, setValue] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setValue(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getTextRecord(name, key, config)\n .then((v) => {\n if (!cancelled) setValue(v);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name, key]);\n\n return { value, loading, error };\n}\n\n/**\n * Avatar URL hook: resolves avatar from text record, handling IPFS and NFT formats\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { url, loading } = useMNSAvatar(name)\n * if (loading) return <div className=\"animate-pulse w-10 h-10 rounded-full bg-gray-700\" />\n * return url ? <img src={url} className=\"w-10 h-10 rounded-full\" /> : null\n * }\n * ```\n */\nexport function useMNSAvatar(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [url, setUrl] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setUrl(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getAvatarUrl(name, config)\n .then((u) => {\n if (!cancelled) setUrl(u);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { url, loading, error };\n}\n\n/**\n * Owner hook: get the owner address of a .mon name\n *\n * @example\n * ```tsx\n * function OwnerInfo({ name }: { name: string }) {\n * const { owner, loading } = useMNSOwner(name)\n * if (loading) return <div>Loading...</div>\n * return <div>Owner: {owner}</div>\n * }\n * ```\n */\nexport function useMNSOwner(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [owner, setOwner] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setOwner(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getOwner(name, config)\n .then((o) => {\n if (!cancelled) setOwner(o);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { owner, loading, error };\n}\n\n/**\n * Resolver hook: get the resolver contract address for a .mon name\n *\n * @example\n * ```tsx\n * function ResolverInfo({ name }: { name: string }) {\n * const { resolver, loading } = useMNSResolver(name)\n * return <div>Resolver: {resolver}</div>\n * }\n * ```\n */\nexport function useMNSResolver(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [resolver, setResolver] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setResolver(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getResolver(name, config)\n .then((r) => {\n if (!cancelled) setResolver(r);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { resolver, loading, error };\n}\n\n/**\n * Expiry hook: get expiration info for a .mon name\n *\n * @example\n * ```tsx\n * function ExpiryInfo({ name }: { name: string }) {\n * const { expiry, expiryDate, daysUntilExpiry, isExpired, loading } = useMNSExpiry(name)\n * if (loading) return <div>Loading...</div>\n * if (isExpired) return <div className=\"text-red-500\">Expired!</div>\n * return <div>Expires in {daysUntilExpiry} days</div>\n * }\n * ```\n */\nexport function useMNSExpiry(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [expiry, setExpiry] = useState<number | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setExpiry(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getExpiry(name, config)\n .then((e) => {\n if (!cancelled) setExpiry(e);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n const expiryDate = expiry ? new Date(expiry * 1000) : null;\n const daysUntilExpiry = expiry\n ? Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n : null;\n const isExpired = expiry ? expiry * 1000 < Date.now() : null;\n\n return { expiry, expiryDate, daysUntilExpiry, isExpired, loading, error };\n}\n\n// Write hooks\nexport {\n useRegistrationInfo,\n useMNSRegister,\n useMNSTextRecords,\n useMNSAddr,\n} from './react-write';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n type RegistrationInfo,\n} from './write';\nimport type { MNSClientConfig } from './client';\nimport { validateAvatarFull, validateMonName } from './utils';\nimport { MAX_AVATAR_BYTES } from './constants';\n\nexport { TEXT_RECORD_KEYS, MAX_AVATAR_BYTES };\nexport type { RegistrationInfo };\n\n/**\n * Hook to get registration info (price, availability, freebie status).\n * Use this to build a registration UI before the user submits.\n *\n * @example\n * ```tsx\n * function RegisterPage() {\n * const [name, setName] = useState('')\n * const { address } = useAccount()\n * const { info, loading, error } = useRegistrationInfo(name, address)\n *\n * return (\n * <div>\n * <input value={name} onChange={(e) => setName(e.target.value)} />\n * {info?.available && (\n * <p>{info.isFreebie ? 'FREE!' : `${formatEther(info.price)} MON`}</p>\n * )}\n * </div>\n * )\n * }\n * ```\n */\nexport function useRegistrationInfo(\n label: string | undefined,\n userAddress: string | undefined,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n const [info, setInfo] = useState<RegistrationInfo | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!label || !userAddress || label.trim().length < 3) {\n setInfo(null);\n setError(null);\n setLoading(false);\n return;\n }\n\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n setInfo(null);\n setError(validation.error || 'Invalid name');\n setLoading(false);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n const timer = setTimeout(() => {\n getRegistrationInfo(label, userAddress, durationYears, config)\n .then((i) => {\n if (!cancelled) setInfo(i);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n }, 300); // debounce\n\n return () => {\n cancelled = true;\n clearTimeout(timer);\n };\n }, [label, userAddress, durationYears, config]);\n\n return { info, loading, error };\n}\n\n// Type for the registration transaction\ntype RegisterTx = Awaited<ReturnType<typeof getRegisterTx>>;\n\n/**\n * Hook to register a .mon name.\n * Returns a prepare function that gives you tx params for wagmi's writeContract.\n *\n * @example\n * ```tsx\n * const { prepare, loading, error } = useMNSRegister()\n * const { writeContract, data: hash } = useWriteContract()\n *\n * const handleRegister = async () => {\n * const tx = await prepare('alice', address)\n * if (tx) writeContract(tx)\n * }\n * ```\n */\nexport function useMNSRegister(config?: MNSClientConfig) {\n const [tx, setTx] = useState<RegisterTx | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const prepare = useCallback(\n async (label: string, ownerAddress: string, durationYears = 1) => {\n setLoading(true);\n setError(null);\n setTx(null);\n\n try {\n const txParams = await getRegisterTx(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n setTx(txParams);\n return txParams;\n } catch (e: any) {\n setError(e.message);\n return null;\n } finally {\n setLoading(false);\n }\n },\n [],\n );\n\n return { prepare, tx, loading, error };\n}\n\n/**\n * Hook to set text records (avatar, url, twitter, etc.)\n *\n * @example\n * ```tsx\n * const { setAvatar, setTwitter } = useMNSTextRecords()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAvatar('alice.mon', 'https://example.com/pic.png'))\n * ```\n */\nexport function useMNSTextRecords() {\n const setTextRecord = useCallback(\n (name: string, key: string, value: string) => {\n return getSetTextTx(name, key, value);\n },\n [],\n );\n\n const setAvatar = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setAvatarValidated = useCallback(async (name: string, url: string) => {\n await validateAvatarFull(url); // Throws if invalid\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setUrl = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'url', url);\n }, []);\n\n const setTwitter = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.twitter', handle);\n }, []);\n\n const setGithub = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.github', handle);\n }, []);\n\n const setDiscord = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.discord', handle);\n }, []);\n\n const setDescription = useCallback((name: string, desc: string) => {\n return getSetTextTx(name, 'description', desc);\n }, []);\n\n return {\n setTextRecord,\n setAvatar,\n setAvatarValidated,\n setUrl,\n setTwitter,\n setGithub,\n setDiscord,\n setDescription,\n textRecordKeys: TEXT_RECORD_KEYS,\n };\n}\n\n/**\n * Hook to update the address a name points to.\n *\n * @example\n * ```tsx\n * const { setAddr } = useMNSAddr()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAddr('alice.mon', '0xnewaddress...'))\n * ```\n */\nexport function useMNSAddr() {\n const setAddr = useCallback((name: string, addr: string) => {\n return getSetAddrTx(name, addr);\n }, []);\n\n return { setAddr };\n}\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 3_000_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":"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","names":["useState","useEffect","useCallback","namehash","namehash","useState","useEffect","useCallback"]}
|
|
1
|
+
{"version":3,"sources":["../src/react.ts","../src/resolve.ts","../src/client.ts","../src/constants.ts","../src/react-write.ts","../src/write.ts","../src/utils.ts"],"sourcesContent":["'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n resolveName,\n lookupAddress,\n getTextRecord,\n getAvatarUrl,\n getOwner,\n getResolver,\n getExpiry,\n} from './resolve';\nimport type { MNSClientConfig } from './client';\n\nexport type { MNSClientConfig };\n\n/**\n * Reverse resolution hook: address → .mon name\n *\n * @example\n * ```tsx\n * function Profile({ address }: { address: string }) {\n * const { name, loading } = useMNSName(address)\n * return <span>{loading ? '...' : name ?? address}</span>\n * }\n * ```\n */\nexport function useMNSName(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!address || !address.startsWith('0x') || address.length !== 42) {\n setName(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n lookupAddress(address, config)\n .then((n) => {\n if (!cancelled) setName(n);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [address]);\n\n return { name, loading, error };\n}\n\n/**\n * Forward resolution hook: .mon name → address\n *\n * @example\n * ```tsx\n * function Lookup() {\n * const { address, loading } = useMNSAddress('alice.mon')\n * return <span>{loading ? 'Resolving...' : address}</span>\n * }\n * ```\n */\nexport function useMNSAddress(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [address, setAddress] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setAddress(null);\n return;\n }\n const fullName = name.endsWith('.mon') ? name : `${name}.mon`;\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n resolveName(fullName, config)\n .then((a) => {\n if (!cancelled) setAddress(a);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { address, loading, error };\n}\n\n/**\n * Display name hook: .mon name if available, otherwise truncated address\n *\n * @example\n * ```tsx\n * function Badge({ address }: { address: string }) {\n * const { displayName, monName } = useMNSDisplay(address)\n * return (\n * <span className={monName ? 'text-purple-400' : 'text-gray-500'}>\n * {displayName}\n * </span>\n * )\n * }\n * ```\n */\nexport function useMNSDisplay(\n address: string | undefined,\n config?: MNSClientConfig,\n) {\n const { name, loading, error } = useMNSName(address, config);\n const truncated = address\n ? `${address.slice(0, 6)}...${address.slice(-4)}`\n : '';\n\n return {\n displayName: name || truncated,\n monName: name,\n loading,\n error,\n };\n}\n\n/**\n * On-demand resolver hook: for input fields accepting names or addresses\n *\n * @example\n * ```tsx\n * function SendForm() {\n * const { resolve, address, name, loading, error } = useMNSResolve()\n * return (\n * <div>\n * <input placeholder=\"Address or .mon name\" onChange={(e) => resolve(e.target.value)} />\n * {loading && <span>Resolving...</span>}\n * {address && <span>→ {address}</span>}\n * </div>\n * )\n * }\n * ```\n */\nexport function useMNSResolve(config?: MNSClientConfig) {\n const [address, setAddress] = useState<string | null>(null);\n const [name, setName] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const resolve = useCallback(async (input: string) => {\n const trimmed = input.trim();\n setError(null);\n\n if (!trimmed) {\n setAddress(null);\n setName(null);\n return;\n }\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) {\n setAddress(trimmed);\n setLoading(true);\n const n = await lookupAddress(trimmed, config);\n setName(n);\n setLoading(false);\n return;\n }\n\n const fullName = trimmed.endsWith('.mon') ? trimmed : `${trimmed}.mon`;\n setName(fullName);\n setLoading(true);\n\n try {\n const addr = await resolveName(fullName, config);\n setAddress(addr);\n if (!addr) setError(`${fullName} not found`);\n } catch (e: any) {\n setError(e.message);\n setAddress(null);\n } finally {\n setLoading(false);\n }\n }, []);\n\n return { resolve, address, name, loading, error };\n}\n\n/**\n * Text record hook: get metadata like avatar, URL, twitter handle\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { value } = useMNSText(name, 'avatar')\n * return value ? <img src={value} /> : null\n * }\n * ```\n */\nexport function useMNSText(\n name: string | undefined,\n key: string,\n config?: MNSClientConfig,\n) {\n const [value, setValue] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setValue(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getTextRecord(name, key, config)\n .then((v) => {\n if (!cancelled) setValue(v);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name, key]);\n\n return { value, loading, error };\n}\n\n/**\n * Avatar URL hook: resolves avatar from text record, handling IPFS and NFT formats\n *\n * @example\n * ```tsx\n * function Avatar({ name }: { name: string }) {\n * const { url, loading } = useMNSAvatar(name)\n * if (loading) return <div className=\"animate-pulse w-10 h-10 rounded-full bg-gray-700\" />\n * return url ? <img src={url} className=\"w-10 h-10 rounded-full\" /> : null\n * }\n * ```\n */\nexport function useMNSAvatar(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [url, setUrl] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setUrl(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getAvatarUrl(name, config)\n .then((u) => {\n if (!cancelled) setUrl(u);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { url, loading, error };\n}\n\n/**\n * Owner hook: get the owner address of a .mon name\n *\n * @example\n * ```tsx\n * function OwnerInfo({ name }: { name: string }) {\n * const { owner, loading } = useMNSOwner(name)\n * if (loading) return <div>Loading...</div>\n * return <div>Owner: {owner}</div>\n * }\n * ```\n */\nexport function useMNSOwner(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [owner, setOwner] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setOwner(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getOwner(name, config)\n .then((o) => {\n if (!cancelled) setOwner(o);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { owner, loading, error };\n}\n\n/**\n * Resolver hook: get the resolver contract address for a .mon name\n *\n * @example\n * ```tsx\n * function ResolverInfo({ name }: { name: string }) {\n * const { resolver, loading } = useMNSResolver(name)\n * return <div>Resolver: {resolver}</div>\n * }\n * ```\n */\nexport function useMNSResolver(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [resolver, setResolver] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setResolver(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getResolver(name, config)\n .then((r) => {\n if (!cancelled) setResolver(r);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n return { resolver, loading, error };\n}\n\n/**\n * Expiry hook: get expiration info for a .mon name\n *\n * @example\n * ```tsx\n * function ExpiryInfo({ name }: { name: string }) {\n * const { expiry, expiryDate, daysUntilExpiry, isExpired, loading } = useMNSExpiry(name)\n * if (loading) return <div>Loading...</div>\n * if (isExpired) return <div className=\"text-red-500\">Expired!</div>\n * return <div>Expires in {daysUntilExpiry} days</div>\n * }\n * ```\n */\nexport function useMNSExpiry(\n name: string | undefined,\n config?: MNSClientConfig,\n) {\n const [expiry, setExpiry] = useState<number | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!name) {\n setExpiry(null);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n getExpiry(name, config)\n .then((e) => {\n if (!cancelled) setExpiry(e);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n\n return () => {\n cancelled = true;\n };\n }, [name]);\n\n const expiryDate = expiry ? new Date(expiry * 1000) : null;\n const daysUntilExpiry = expiry\n ? Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n : null;\n const isExpired = expiry ? expiry * 1000 < Date.now() : null;\n\n return { expiry, expiryDate, daysUntilExpiry, isExpired, loading, error };\n}\n\n// Write hooks\nexport {\n useRegistrationInfo,\n useMNSRegister,\n useMNSTextRecords,\n useMNSAddr,\n} from './react-write';\n","import { namehash, keccak256, toBytes } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport {\n MNS_REGISTRY,\n MNS_BASE_REGISTRAR,\n ZERO_ADDRESS,\n registryAbi,\n resolverAbi,\n} from './constants';\n\n// ─── Cache ─────────────────────────────────────────────────────\n\nconst nameCache = new Map<string, { value: string | null; ts: number }>();\nconst addrCache = new Map<string, { value: string | null; ts: number }>();\n\nconst CACHE_TTL = 60_000; // 1 minute\n\nfunction getCached(\n cache: Map<string, { value: string | null; ts: number }>,\n key: string,\n) {\n const entry = cache.get(key);\n if (entry && Date.now() - entry.ts < CACHE_TTL) return entry.value;\n return undefined;\n}\n\n// ─── Forward Resolution ────────────────────────────────────────\n\n/**\n * Resolve a .mon name to an address.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The resolved address, or null if not found\n *\n * @example\n * ```ts\n * import { resolveName } from '@monadns/sdk'\n *\n * const address = await resolveName('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function resolveName(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = name.toLowerCase();\n const cached = getCached(addrCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const node = namehash(key) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const addr = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'addr',\n args: [node],\n });\n\n const result = addr === ZERO_ADDRESS ? null : addr;\n addrCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n addrCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Reverse Resolution ────────────────────────────────────────\n\n/**\n * Look up the primary .mon name for an address.\n *\n * @param address - A 0x-prefixed Monad address\n * @param config - Optional client configuration\n * @returns The primary name (e.g. \"alice.mon\"), or null\n *\n * @example\n * ```ts\n * import { lookupAddress } from '@monadns/sdk'\n *\n * const name = await lookupAddress('0xa05a8BF1...')\n * // 'alice.mon'\n * ```\n */\nexport async function lookupAddress(\n address: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const key = address.toLowerCase();\n const cached = getCached(nameCache, key);\n if (cached !== undefined) return cached;\n\n try {\n const client = getMNSClient(config);\n const reverseNode = namehash(\n `${key.slice(2)}.addr.reverse`,\n ) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [reverseNode],\n });\n\n if (resolver === ZERO_ADDRESS) {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n\n const name = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'name',\n args: [reverseNode],\n });\n\n const result = name || null;\n nameCache.set(key, { value: result, ts: Date.now() });\n return result;\n } catch {\n nameCache.set(key, { value: null, ts: Date.now() });\n return null;\n }\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get a text record for a .mon name.\n *\n * @param name - Full name including .mon\n * @param key - The text record key (e.g. \"avatar\", \"url\", \"com.twitter\")\n * @param config - Optional client configuration\n *\n * @example\n * ```ts\n * const avatar = await getTextRecord('alice.mon', 'avatar')\n * const twitter = await getTextRecord('alice.mon', 'com.twitter')\n * ```\n */\nexport async function getTextRecord(\n name: string,\n key: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n if (resolver === ZERO_ADDRESS) return null;\n\n const value = await client.readContract({\n address: resolver,\n abi: resolverAbi,\n functionName: 'text',\n args: [node, key],\n });\n\n return value || null;\n } catch {\n return null;\n }\n}\n\n// ─── Ownership & Registry Info ─────────────────────────────────\n\n/**\n * Get the owner of a .mon name.\n *\n * @param name - Full name including .mon (e.g. \"alice.mon\")\n * @param config - Optional client configuration\n * @returns The owner address, or null if not registered\n *\n * @example\n * ```ts\n * const owner = await getOwner('alice.mon')\n * // '0xa05a8BF1eda5bbC2b3aCAF03D04f77bD7d66Cc47'\n * ```\n */\nexport async function getOwner(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const owner = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'owner',\n args: [node],\n });\n\n return owner === ZERO_ADDRESS ? null : owner;\n } catch {\n return null;\n }\n}\n\n/**\n * Get the resolver address for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns The resolver contract address, or null if not set\n *\n * @example\n * ```ts\n * const resolver = await getResolver('alice.mon')\n * // '0xa2eb94c88e55d944aced2066c5cec9b759801f97'\n * ```\n */\nexport async function getResolver(\n name: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n try {\n const client = getMNSClient(config);\n const node = namehash(name.toLowerCase()) as `0x${string}`;\n\n const resolver = await client.readContract({\n address: MNS_REGISTRY,\n abi: registryAbi,\n functionName: 'resolver',\n args: [node],\n });\n\n return resolver === ZERO_ADDRESS ? null : resolver;\n } catch {\n return null;\n }\n}\n\n/**\n * Get expiry timestamp for a .mon name.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns Unix timestamp (seconds) when the name expires, or null if not found\n *\n * @example\n * ```ts\n * const expiry = await getExpiry('alice.mon')\n * // 1735689600 (Unix timestamp)\n *\n * const expiryDate = new Date(expiry * 1000)\n * const daysLeft = Math.floor((expiry * 1000 - Date.now()) / (1000 * 60 * 60 * 24))\n * ```\n */\nexport async function getExpiry(\n name: string,\n config?: MNSClientConfig,\n): Promise<number | null> {\n try {\n const client = getMNSClient(config);\n const label = name.replace('.mon', '');\n const labelHash = keccak256(toBytes(label));\n\n const expiry = await client.readContract({\n address: MNS_BASE_REGISTRAR,\n abi: [\n {\n name: 'nameExpires',\n type: 'function',\n inputs: [{ name: 'id', type: 'uint256' }],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n ] as const,\n functionName: 'nameExpires',\n args: [BigInt(labelHash)],\n });\n\n return Number(expiry);\n } catch {\n return null;\n }\n}\n\n/**\n * Check if a .mon name is available for registration.\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @returns true if available (not registered), false if taken\n *\n * @example\n * ```ts\n * const available = await getAvailable('alice.mon')\n * if (available) {\n * console.log('alice.mon is available!')\n * }\n * ```\n */\nexport async function getAvailable(\n name: string,\n config?: MNSClientConfig,\n): Promise<boolean> {\n const addr = await resolveName(name, config);\n return addr === null;\n}\n\n// ─── Helpers ───────────────────────────────────────────────────\n\n/**\n * Get a display-friendly name for an address.\n * Returns the .mon name if set, otherwise a truncated address.\n *\n * @example\n * ```ts\n * const display = await getDisplayName('0xa05a8BF1...')\n * // 'alice.mon' or '0xa05a...Cc47'\n * ```\n */\nexport async function getDisplayName(\n address: string,\n config?: MNSClientConfig,\n): Promise<string> {\n const name = await lookupAddress(address, config);\n if (name) return name;\n return `${address.slice(0, 6)}...${address.slice(-4)}`;\n}\n\n/**\n * Smart resolver — accepts either a .mon name or a raw address.\n * Useful for input fields where users can type either.\n *\n * @returns The resolved address, or null if invalid/not found\n *\n * @example\n * ```ts\n * const addr = await resolveInput('alice.mon') // resolves name\n * const addr = await resolveInput('alice') // tries alice.mon\n * const addr = await resolveInput('0xa05a...') // passes through\n * ```\n */\nexport async function resolveInput(\n input: string,\n config?: MNSClientConfig,\n): Promise<string | null> {\n const trimmed = input.trim();\n if (!trimmed) return null;\n\n if (trimmed.startsWith('0x') && trimmed.length === 42) return trimmed;\n if (trimmed.endsWith('.mon')) return resolveName(trimmed, config);\n\n // Try appending .mon\n return resolveName(`${trimmed}.mon`, config);\n}\n\n/**\n * Get a resolved avatar URL for a .mon name.\n *\n * Handles multiple avatar formats:\n * - Direct HTTPS URLs: returned as-is\n * - IPFS URIs (ipfs://...): resolved via public gateway\n * - Raw IPFS CIDs (Qm..., bafy...): converted to IPFS gateway URLs\n * - Arweave URIs (ar://...): resolved via arweave.net\n * - NFT references (eip155:chainId/erc721:contract/tokenId): fetches tokenURI and extracts image\n * - Data URIs: returned as-is\n *\n * @param name - Full name including .mon\n * @param config - Optional client configuration\n * @param options - Optional fallback URL to return if avatar not found\n * @returns A usable HTTPS URL, or null (or fallback)\n *\n * @example\n * ```ts\n * const avatarUrl = await getAvatarUrl('alice.mon')\n * // 'https://example.com/avatar.png'\n *\n * // With fallback\n * const avatarUrl = await getAvatarUrl('alice.mon', undefined, {\n * fallback: DEFAULT_AVATAR_PLACEHOLDER\n * })\n * ```\n */\nexport async function getAvatarUrl(\n name: string,\n config?: MNSClientConfig,\n options?: { fallback?: string },\n): Promise<string | null> {\n const raw = await getTextRecord(name, 'avatar', config);\n if (!raw) return options?.fallback || null;\n\n try {\n // Direct HTTP(S) URL\n if (raw.startsWith('http://') || raw.startsWith('https://')) {\n return raw;\n }\n\n // IPFS with protocol\n if (raw.startsWith('ipfs://')) {\n const hash = raw.slice(7);\n return `https://ipfs.io/ipfs/${hash}`;\n }\n\n // Raw IPFS CID (Qm... or bafy...)\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(raw)) {\n return `https://ipfs.io/ipfs/${raw}`;\n }\n\n // Arweave\n if (raw.startsWith('ar://')) {\n const hash = raw.slice(5);\n return `https://arweave.net/${hash}`;\n }\n\n // NFT reference: eip155:{chainId}/erc721:{contract}/{tokenId}\n // or eip155:{chainId}/erc1155:{contract}/{tokenId}\n const nftMatch = raw.match(\n /^eip155:(\\d+)\\/(erc721|erc1155):0x([a-fA-F0-9]{40})\\/(\\d+)$/,\n );\n if (nftMatch) {\n const client = getMNSClient(config);\n const contract = `0x${nftMatch[3]}` as `0x${string}`;\n const tokenId = BigInt(nftMatch[4]);\n\n const tokenUriAbi = [\n {\n name: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n type: 'function',\n inputs: [{ name: 'tokenId', type: 'uint256' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n ] as const;\n\n let tokenUri = (await client.readContract({\n address: contract,\n abi: tokenUriAbi,\n functionName: nftMatch[2] === 'erc721' ? 'tokenURI' : 'uri',\n args: [tokenId],\n })) as string;\n\n // Resolve IPFS tokenURI\n if (tokenUri.startsWith('ipfs://')) {\n tokenUri = `https://ipfs.io/ipfs/${tokenUri.slice(7)}`;\n }\n\n // Fetch metadata JSON and extract image\n const res = await fetch(tokenUri);\n const metadata = await res.json();\n let image = metadata.image || metadata.image_url || null;\n\n if (image && image.startsWith('ipfs://')) {\n image = `https://ipfs.io/ipfs/${image.slice(7)}`;\n }\n\n return image || options?.fallback || null;\n }\n\n // data: URIs\n if (raw.startsWith('data:')) {\n return raw;\n }\n\n return options?.fallback || null;\n } catch (error) {\n console.warn('Failed to resolve avatar:', error);\n return options?.fallback || null;\n }\n}\n\n/**\n * Clear the internal resolution cache.\n * Useful after a new registration.\n */\nexport function clearCache() {\n nameCache.clear();\n addrCache.clear();\n}\n","import {\n createPublicClient,\n http,\n type PublicClient,\n type Chain,\n type Transport,\n} from 'viem';\nimport { monad } from 'viem/chains';\n\nexport type MNSClientConfig = {\n /** Custom RPC URL. Defaults to the public Monad RPC. */\n rpcUrl?: string;\n /** Existing viem PublicClient to reuse instead of creating a new one. */\n client?: PublicClient<Transport, Chain>;\n};\n\nlet defaultClient: PublicClient<Transport, Chain> | null = null;\n\n/**\n * Get or create the MNS public client.\n * Reuses a singleton by default; pass config to override.\n */\nexport function getMNSClient(\n config?: MNSClientConfig,\n): PublicClient<Transport, Chain> {\n if (config?.client) return config.client;\n\n if (config?.rpcUrl) {\n return createPublicClient({\n chain: monad,\n transport: http(config.rpcUrl),\n });\n }\n\n if (!defaultClient) {\n defaultClient = createPublicClient({\n chain: monad,\n transport: http(),\n });\n }\n\n return defaultClient;\n}\n","export const MNS_REGISTRY =\n '0x13f963486e741c8d3fcdc0a34a910920339a19ce' as const;\nexport const MNS_PUBLIC_RESOLVER =\n '0xa2eb94c88e55d944aced2066c5cec9b759801f97' as const;\nexport const MNS_CONTROLLER =\n '0x98866c55adbc73ec6c272bb3604ddbdee3f282a8' as const;\nexport const MNS_BASE_REGISTRAR =\n '0x104a49db9318c284d462841b6940bdb46624ca55' as const;\n\nexport const ZERO_ADDRESS =\n '0x0000000000000000000000000000000000000000' as const;\n\n/**\n * Maximum recommended avatar size in bytes (50KB)\n * Strictly enforced for data URIs, recommended for all avatars\n */\nexport const MAX_AVATAR_BYTES = 51_200; // 50KB\n\n/**\n * Default avatar placeholder (1x1 transparent SVG)\n */\nexport const DEFAULT_AVATAR_PLACEHOLDER =\n 'data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\"%3E%3C/svg%3E';\n\nexport const registryAbi = [\n {\n name: 'resolver',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'owner',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const resolverAbi = [\n {\n name: 'addr',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'address' }],\n stateMutability: 'view',\n },\n {\n name: 'name',\n type: 'function',\n inputs: [{ name: 'node', type: 'bytes32' }],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n {\n name: 'text',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n ],\n outputs: [{ name: '', type: 'string' }],\n stateMutability: 'view',\n },\n] as const;\n","'use client';\n\nimport { useState, useEffect, useCallback } from 'react';\nimport {\n getRegistrationInfo,\n getRegisterTx,\n getSetTextTx,\n getSetAddrTx,\n TEXT_RECORD_KEYS,\n type RegistrationInfo,\n} from './write';\nimport type { MNSClientConfig } from './client';\nimport { validateAvatarFull, validateMonName } from './utils';\nimport { MAX_AVATAR_BYTES } from './constants';\n\nexport { TEXT_RECORD_KEYS, MAX_AVATAR_BYTES };\nexport type { RegistrationInfo };\n\n/**\n * Hook to get registration info (price, availability, freebie status).\n * Use this to build a registration UI before the user submits.\n *\n * @example\n * ```tsx\n * function RegisterPage() {\n * const [name, setName] = useState('')\n * const { address } = useAccount()\n * const { info, loading, error } = useRegistrationInfo(name, address)\n *\n * return (\n * <div>\n * <input value={name} onChange={(e) => setName(e.target.value)} />\n * {info?.available && (\n * <p>{info.isFreebie ? 'FREE!' : `${formatEther(info.price)} MON`}</p>\n * )}\n * </div>\n * )\n * }\n * ```\n */\nexport function useRegistrationInfo(\n label: string | undefined,\n userAddress: string | undefined,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n const [info, setInfo] = useState<RegistrationInfo | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n useEffect(() => {\n if (!label || !userAddress || label.trim().length < 3) {\n setInfo(null);\n setError(null);\n setLoading(false);\n return;\n }\n\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n setInfo(null);\n setError(validation.error || 'Invalid name');\n setLoading(false);\n return;\n }\n\n let cancelled = false;\n setLoading(true);\n setError(null);\n\n const timer = setTimeout(() => {\n getRegistrationInfo(label, userAddress, durationYears, config)\n .then((i) => {\n if (!cancelled) setInfo(i);\n })\n .catch((e) => {\n if (!cancelled) setError(e.message);\n })\n .finally(() => {\n if (!cancelled) setLoading(false);\n });\n }, 300); // debounce\n\n return () => {\n cancelled = true;\n clearTimeout(timer);\n };\n }, [label, userAddress, durationYears, config]);\n\n return { info, loading, error };\n}\n\n// Type for the registration transaction\ntype RegisterTx = Awaited<ReturnType<typeof getRegisterTx>>;\n\n/**\n * Hook to register a .mon name.\n * Returns a prepare function that gives you tx params for wagmi's writeContract.\n *\n * @example\n * ```tsx\n * const { prepare, loading, error } = useMNSRegister()\n * const { writeContract, data: hash } = useWriteContract()\n *\n * const handleRegister = async () => {\n * const tx = await prepare('alice', address)\n * if (tx) writeContract(tx)\n * }\n * ```\n */\nexport function useMNSRegister(config?: MNSClientConfig) {\n const [tx, setTx] = useState<RegisterTx | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const prepare = useCallback(\n async (label: string, ownerAddress: string, durationYears = 1) => {\n setLoading(true);\n setError(null);\n setTx(null);\n\n try {\n const txParams = await getRegisterTx(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n setTx(txParams);\n return txParams;\n } catch (e: any) {\n setError(e.message);\n return null;\n } finally {\n setLoading(false);\n }\n },\n [],\n );\n\n return { prepare, tx, loading, error };\n}\n\n/**\n * Hook to set text records (avatar, url, twitter, etc.)\n *\n * @example\n * ```tsx\n * const { setAvatar, setTwitter } = useMNSTextRecords()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAvatar('alice.mon', 'https://example.com/pic.png'))\n * ```\n */\nexport function useMNSTextRecords() {\n const setTextRecord = useCallback(\n (name: string, key: string, value: string) => {\n return getSetTextTx(name, key, value);\n },\n [],\n );\n\n const setAvatar = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setAvatarValidated = useCallback(async (name: string, url: string) => {\n await validateAvatarFull(url); // Throws if invalid\n return getSetTextTx(name, 'avatar', url);\n }, []);\n\n const setUrl = useCallback((name: string, url: string) => {\n return getSetTextTx(name, 'url', url);\n }, []);\n\n const setTwitter = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.twitter', handle);\n }, []);\n\n const setGithub = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.github', handle);\n }, []);\n\n const setDiscord = useCallback((name: string, handle: string) => {\n return getSetTextTx(name, 'com.discord', handle);\n }, []);\n\n const setDescription = useCallback((name: string, desc: string) => {\n return getSetTextTx(name, 'description', desc);\n }, []);\n\n return {\n setTextRecord,\n setAvatar,\n setAvatarValidated,\n setUrl,\n setTwitter,\n setGithub,\n setDiscord,\n setDescription,\n textRecordKeys: TEXT_RECORD_KEYS,\n };\n}\n\n/**\n * Hook to update the address a name points to.\n *\n * @example\n * ```tsx\n * const { setAddr } = useMNSAddr()\n * const { writeContract } = useWriteContract()\n *\n * writeContract(setAddr('alice.mon', '0xnewaddress...'))\n * ```\n */\nexport function useMNSAddr() {\n const setAddr = useCallback((name: string, addr: string) => {\n return getSetAddrTx(name, addr);\n }, []);\n\n return { setAddr };\n}\n","import { namehash } from 'viem';\nimport { getMNSClient, type MNSClientConfig } from './client';\nimport { MNS_CONTROLLER, MNS_PUBLIC_RESOLVER } from './constants';\nimport { validateAvatarUri, validateMonName } from './utils';\n\n/**\n * MNS Write Helpers\n *\n * These return transaction parameters ({ address, abi, functionName, args, value })\n * that you pass to wagmi's writeContract or any wallet client.\n * The user signs with their own wallet.\n */\n\n// ─── ABIs ──────────────────────────────────────────────────────\n\nexport const controllerReadAbi = [\n {\n name: 'available',\n type: 'function',\n inputs: [{ name: 'label', type: 'string' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n {\n name: 'rentPrice',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'view',\n },\n {\n name: 'hasClaimedFreebie',\n type: 'function',\n inputs: [{ name: 'user', type: 'address' }],\n outputs: [{ name: '', type: 'bool' }],\n stateMutability: 'view',\n },\n] as const;\n\nexport const controllerWriteAbi = [\n {\n name: 'register',\n type: 'function',\n inputs: [\n { name: 'label', type: 'string' },\n { name: 'owner', type: 'address' },\n { name: 'duration', type: 'uint256' },\n ],\n outputs: [{ name: '', type: 'uint256' }],\n stateMutability: 'payable',\n },\n] as const;\n\nexport const resolverWriteAbi = [\n {\n name: 'setText',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'key', type: 'string' },\n { name: 'value', type: 'string' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n {\n name: 'setAddr',\n type: 'function',\n inputs: [\n { name: 'node', type: 'bytes32' },\n { name: 'a', type: 'address' },\n ],\n outputs: [],\n stateMutability: 'nonpayable',\n },\n] as const;\n\n// ─── Types ─────────────────────────────────────────────────────\n\nexport type RegistrationInfo = {\n available: boolean;\n price: bigint;\n isFreebie: boolean;\n hasClaimed: boolean;\n duration: bigint;\n label: string;\n};\n\n// ─── Registration ──────────────────────────────────────────────\n\nconst ONE_YEAR = BigInt(365 * 24 * 60 * 60);\n\n/**\n * Get all info needed before registration.\n *\n * @example\n * ```ts\n * const info = await getRegistrationInfo('alice', userAddress)\n * // { available: true, price: 96831564n..., isFreebie: true, ... }\n * ```\n */\nexport async function getRegistrationInfo(\n label: string,\n userAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n): Promise<RegistrationInfo> {\n // Validate name format before making RPC calls\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const client = getMNSClient(config);\n const duration = ONE_YEAR * BigInt(durationYears);\n\n const [available, price, hasClaimed, balance] = await Promise.all([\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'available',\n args: [label],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'rentPrice',\n args: [label, duration],\n }),\n client.readContract({\n address: MNS_CONTROLLER,\n abi: controllerReadAbi,\n functionName: 'hasClaimedFreebie',\n args: [userAddress as `0x${string}`],\n }),\n client.getBalance({ address: userAddress as `0x${string}` }),\n ]);\n\n const isFreebie = balance >= 10n * 10n ** 18n && !hasClaimed;\n\n return { available, price, isFreebie, hasClaimed, duration, label };\n}\n\n/**\n * Get transaction parameters for registering a .mon name.\n * Pass the result to wagmi's writeContract.\n *\n * @example\n * ```ts\n * const tx = await getRegisterTx('alice', userAddress)\n * writeContract(tx)\n * ```\n */\nexport async function getRegisterTx(\n label: string,\n ownerAddress: string,\n durationYears = 1,\n config?: MNSClientConfig,\n) {\n // Validate name format\n const validation = validateMonName(label);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n\n const info = await getRegistrationInfo(\n label,\n ownerAddress,\n durationYears,\n config,\n );\n\n if (!info.available) {\n throw new Error(`${label}.mon is not available`);\n }\n\n return {\n address: MNS_CONTROLLER,\n abi: controllerWriteAbi,\n functionName: 'register' as const,\n args: [label, ownerAddress as `0x${string}`, info.duration] as const,\n value: info.isFreebie ? 0n : info.price,\n gas: 800_000n,\n _meta: {\n label,\n isFreebie: info.isFreebie,\n price: info.price,\n duration: info.duration,\n },\n };\n}\n\n// ─── Text Records ──────────────────────────────────────────────\n\n/**\n * Get transaction parameters for setting a text record.\n * The connected user must own the name.\n * For avatar records, validates data URI size constraints.\n *\n * @example\n * ```ts\n * const tx = getSetTextTx('alice.mon', 'avatar', 'https://example.com/pic.png')\n * writeContract(tx)\n * ```\n */\nexport function getSetTextTx(name: string, key: string, value: string) {\n // Enforce strict validation for avatar data URIs\n if (key === 'avatar' && value.startsWith('data:')) {\n const validation = validateAvatarUri(value);\n if (!validation.valid) {\n throw new Error(validation.error);\n }\n }\n\n // Log warning for non-validated remote avatars\n if (key === 'avatar' && !value.startsWith('data:')) {\n console.warn(\n '⚠️ Avatar set without size validation. For best practices, use setAvatarValidated() or validateAvatarFull() before calling setAvatar().',\n );\n }\n\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setText' as const,\n args: [node, key, value] as const,\n };\n}\n\n/**\n * Get transaction parameters for setting the address record.\n */\nexport function getSetAddrTx(name: string, addr: string) {\n const node = namehash(\n name.endsWith('.mon') ? name : `${name}.mon`,\n ) as `0x${string}`;\n\n return {\n address: MNS_PUBLIC_RESOLVER,\n abi: resolverWriteAbi,\n functionName: 'setAddr' as const,\n args: [node, addr as `0x${string}`] as const,\n };\n}\n\nexport const TEXT_RECORD_KEYS = {\n avatar: 'avatar',\n url: 'url',\n description: 'description',\n email: 'email',\n twitter: 'com.twitter',\n github: 'com.github',\n discord: 'com.discord',\n telegram: 'org.telegram',\n} as const;\n","import { MAX_AVATAR_BYTES } from './constants';\n\n/**\n * Validate a .mon name according to MNS rules\n *\n * @example\n * ```ts\n * const validation = validateMonName('alice');\n * if (!validation.valid) {\n * console.error(validation.error);\n * }\n * ```\n */\nexport function validateMonName(label: string): {\n valid: boolean;\n error?: string;\n} {\n if (!label || label.trim().length === 0) {\n return { valid: false, error: 'Name cannot be empty' };\n }\n\n const trimmed = label.trim().toLowerCase();\n\n // Cannot start with 0x (looks like an address)\n if (trimmed.startsWith('0x')) {\n return {\n valid: false,\n error: 'Name cannot start with \"0x\" (reserved for addresses)',\n };\n }\n\n // Check for valid characters (alphanumeric + hyphen)\n if (!/^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(trimmed)) {\n return {\n valid: false,\n error:\n 'Name can only contain lowercase letters, numbers, and hyphens (not at start/end)',\n };\n }\n\n // No consecutive hyphens\n if (trimmed.includes('--')) {\n return { valid: false, error: 'Name cannot contain consecutive hyphens' };\n }\n\n // Maximum length (ENS standard is 63 characters for a label)\n if (trimmed.length > 63) {\n return { valid: false, error: 'Name must be 63 characters or less' };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar URI format and size (for data URIs)\n * This performs synchronous validation only\n */\nexport function validateAvatarUri(uri: string): {\n valid: boolean;\n error?: string;\n sizeBytes?: number;\n} {\n if (!uri) {\n return { valid: false, error: 'Avatar URI is required' };\n }\n\n // STRICT: Data URIs go on-chain, must be limited\n if (uri.startsWith('data:')) {\n const sizeBytes = new Blob([uri]).size;\n\n if (sizeBytes > MAX_AVATAR_BYTES) {\n return {\n valid: false,\n error: `Avatar size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${MAX_AVATAR_BYTES / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n sizeBytes,\n };\n }\n\n return { valid: true, sizeBytes };\n }\n\n // For IPFS/HTTP, validate format only (size check is async)\n const validSchemes = ['http://', 'https://', 'ipfs://', 'eip155:', 'ar://'];\n const hasValidScheme = validSchemes.some((scheme) => uri.startsWith(scheme));\n\n if (!hasValidScheme) {\n // Check if it's a raw IPFS CID\n if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n return { valid: true };\n }\n\n return {\n valid: false,\n error: 'Avatar must be a valid HTTP, IPFS, Arweave, or NFT URI',\n };\n }\n\n return { valid: true };\n}\n\n/**\n * Validate avatar including remote file size check\n * Use this in UI before calling setAvatar() for comprehensive validation\n *\n * @throws Error if avatar is invalid or too large\n *\n * @example\n * ```ts\n * try {\n * await validateAvatarFull(avatarUrl);\n * // Safe to use\n * } catch (error) {\n * console.error(error.message);\n * // \"Avatar file size (125.3KB) exceeds 50KB limit...\"\n * }\n * ```\n */\nexport async function validateAvatarFull(\n uri: string,\n options?: { maxBytes?: number; timeout?: number },\n): Promise<{ valid: true; sizeBytes?: number }> {\n const maxBytes = options?.maxBytes || MAX_AVATAR_BYTES;\n const timeout = options?.timeout || 5000;\n\n // First, validate format\n const formatValidation = validateAvatarUri(uri);\n if (!formatValidation.valid) {\n throw new Error(formatValidation.error);\n }\n\n // Data URIs already validated\n if (uri.startsWith('data:')) {\n return { valid: true, sizeBytes: formatValidation.sizeBytes };\n }\n\n // NFT avatars - can't pre-validate size easily\n if (uri.startsWith('eip155:')) {\n return { valid: true };\n }\n\n // For HTTP/IPFS, try to check size via HEAD request\n let checkUrl = uri;\n\n // Convert IPFS to gateway URL for size check\n if (uri.startsWith('ipfs://')) {\n checkUrl = `https://ipfs.io/ipfs/${uri.slice(7)}`;\n } else if (/^(Qm[1-9A-HJ-NP-Za-km-z]{44}|bafy[0-9A-Za-z]{50,})/.test(uri)) {\n checkUrl = `https://ipfs.io/ipfs/${uri}`;\n } else if (uri.startsWith('ar://')) {\n checkUrl = `https://arweave.net/${uri.slice(5)}`;\n }\n\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(checkUrl, {\n method: 'HEAD',\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n const contentLength = response.headers.get('content-length');\n if (contentLength) {\n const sizeBytes = parseInt(contentLength, 10);\n\n if (sizeBytes > maxBytes) {\n throw new Error(\n `Avatar file size (${(sizeBytes / 1024).toFixed(1)}KB) exceeds ${maxBytes / 1024}KB limit. Please optimize as WebP or SVG for better performance.`,\n );\n }\n\n return { valid: true, sizeBytes };\n }\n\n // No content-length header - warn but allow\n console.warn(\n `Could not determine avatar size for ${uri}. Ensure it's under ${maxBytes / 1024}KB.`,\n );\n return { valid: true };\n } catch (error: any) {\n if (error.name === 'AbortError') {\n console.warn(`Avatar size check timed out for ${uri}`);\n return { valid: true };\n }\n\n // If it's our size error, re-throw it\n if (error.message.includes('exceeds')) {\n throw error;\n }\n\n // Network error - warn but don't block\n console.warn(`Could not check avatar size: ${error.message}`);\n return { valid: true };\n }\n}\n\n/**\n * Get file size of a remote URL (helper for UI validation)\n * Returns size in bytes or null if unable to determine\n *\n * @example\n * ```ts\n * const size = await getRemoteAvatarSize('https://example.com/avatar.png');\n * if (size && size > MAX_AVATAR_BYTES) {\n * alert('Avatar too large!');\n * }\n * ```\n */\nexport async function getRemoteAvatarSize(url: string): Promise<number | null> {\n try {\n const response = await fetch(url, { method: 'HEAD' });\n const contentLength = response.headers.get('content-length');\n return contentLength ? parseInt(contentLength, 10) : null;\n } catch {\n return null;\n }\n}\n"],"mappings":";;;;AAEA,SAAS,YAAAA,WAAU,aAAAC,YAAW,eAAAC,oBAAmB;;;ACFjD,SAAS,UAAU,WAAW,eAAe;;;ACA7C;AAAA,EACE;AAAA,EACA;AAAA,OAIK;AACP,SAAS,aAAa;AAStB,IAAI,gBAAuD;AAMpD,SAAS,aACd,QACgC;AAChC,MAAI,QAAQ,OAAQ,QAAO,OAAO;AAElC,MAAI,QAAQ,QAAQ;AAClB,WAAO,mBAAmB;AAAA,MACxB,OAAO;AAAA,MACP,WAAW,KAAK,OAAO,MAAM;AAAA,IAC/B,CAAC;AAAA,EACH;AAEA,MAAI,CAAC,eAAe;AAClB,oBAAgB,mBAAmB;AAAA,MACjC,OAAO;AAAA,MACP,WAAW,KAAK;AAAA,IAClB,CAAC;AAAA,EACH;AAEA,SAAO;AACT;;;AC1CO,IAAM,eACX;AACK,IAAM,sBACX;AACK,IAAM,iBACX;AACK,IAAM,qBACX;AAEK,IAAM,eACX;AAMK,IAAM,mBAAmB;AAQzB,IAAM,cAAc;AAAA,EACzB;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,QAAQ,MAAM,UAAU,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,IACvC,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,QAAQ,MAAM,UAAU,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,IACvC,iBAAiB;AAAA,EACnB;AACF;AAEO,IAAM,cAAc;AAAA,EACzB;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,QAAQ,MAAM,UAAU,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,IACvC,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,QAAQ,MAAM,UAAU,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,SAAS,CAAC;AAAA,IACtC,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ;AAAA,MACN,EAAE,MAAM,QAAQ,MAAM,UAAU;AAAA,MAChC,EAAE,MAAM,OAAO,MAAM,SAAS;AAAA,IAChC;AAAA,IACA,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,SAAS,CAAC;AAAA,IACtC,iBAAiB;AAAA,EACnB;AACF;;;AFtDA,IAAM,YAAY,oBAAI,IAAkD;AACxE,IAAM,YAAY,oBAAI,IAAkD;AAExE,IAAM,YAAY;AAElB,SAAS,UACP,OACA,KACA;AACA,QAAM,QAAQ,MAAM,IAAI,GAAG;AAC3B,MAAI,SAAS,KAAK,IAAI,IAAI,MAAM,KAAK,UAAW,QAAO,MAAM;AAC7D,SAAO;AACT;AAmBA,eAAsB,YACpB,MACA,QACwB;AACxB,QAAM,MAAM,KAAK,YAAY;AAC7B,QAAM,SAAS,UAAU,WAAW,GAAG;AACvC,MAAI,WAAW,OAAW,QAAO;AAEjC,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,OAAO,SAAS,GAAG;AAEzB,UAAM,WAAW,MAAM,OAAO,aAAa;AAAA,MACzC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,IAAI;AAAA,IACb,CAAC;AAED,QAAI,aAAa,cAAc;AAC7B,gBAAU,IAAI,KAAK,EAAE,OAAO,MAAM,IAAI,KAAK,IAAI,EAAE,CAAC;AAClD,aAAO;AAAA,IACT;AAEA,UAAM,OAAO,MAAM,OAAO,aAAa;AAAA,MACrC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,IAAI;AAAA,IACb,CAAC;AAED,UAAM,SAAS,SAAS,eAAe,OAAO;AAC9C,cAAU,IAAI,KAAK,EAAE,OAAO,QAAQ,IAAI,KAAK,IAAI,EAAE,CAAC;AACpD,WAAO;AAAA,EACT,QAAQ;AACN,cAAU,IAAI,KAAK,EAAE,OAAO,MAAM,IAAI,KAAK,IAAI,EAAE,CAAC;AAClD,WAAO;AAAA,EACT;AACF;AAmBA,eAAsB,cACpB,SACA,QACwB;AACxB,QAAM,MAAM,QAAQ,YAAY;AAChC,QAAM,SAAS,UAAU,WAAW,GAAG;AACvC,MAAI,WAAW,OAAW,QAAO;AAEjC,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,cAAc;AAAA,MAClB,GAAG,IAAI,MAAM,CAAC,CAAC;AAAA,IACjB;AAEA,UAAM,WAAW,MAAM,OAAO,aAAa;AAAA,MACzC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,WAAW;AAAA,IACpB,CAAC;AAED,QAAI,aAAa,cAAc;AAC7B,gBAAU,IAAI,KAAK,EAAE,OAAO,MAAM,IAAI,KAAK,IAAI,EAAE,CAAC;AAClD,aAAO;AAAA,IACT;AAEA,UAAM,OAAO,MAAM,OAAO,aAAa;AAAA,MACrC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,WAAW;AAAA,IACpB,CAAC;AAED,UAAM,SAAS,QAAQ;AACvB,cAAU,IAAI,KAAK,EAAE,OAAO,QAAQ,IAAI,KAAK,IAAI,EAAE,CAAC;AACpD,WAAO;AAAA,EACT,QAAQ;AACN,cAAU,IAAI,KAAK,EAAE,OAAO,MAAM,IAAI,KAAK,IAAI,EAAE,CAAC;AAClD,WAAO;AAAA,EACT;AACF;AAiBA,eAAsB,cACpB,MACA,KACA,QACwB;AACxB,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,OAAO,SAAS,KAAK,YAAY,CAAC;AAExC,UAAM,WAAW,MAAM,OAAO,aAAa;AAAA,MACzC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,IAAI;AAAA,IACb,CAAC;AAED,QAAI,aAAa,aAAc,QAAO;AAEtC,UAAM,QAAQ,MAAM,OAAO,aAAa;AAAA,MACtC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,MAAM,GAAG;AAAA,IAClB,CAAC;AAED,WAAO,SAAS;AAAA,EAClB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAiBA,eAAsB,SACpB,MACA,QACwB;AACxB,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,OAAO,SAAS,KAAK,YAAY,CAAC;AAExC,UAAM,QAAQ,MAAM,OAAO,aAAa;AAAA,MACtC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,IAAI;AAAA,IACb,CAAC;AAED,WAAO,UAAU,eAAe,OAAO;AAAA,EACzC,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAeA,eAAsB,YACpB,MACA,QACwB;AACxB,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,OAAO,SAAS,KAAK,YAAY,CAAC;AAExC,UAAM,WAAW,MAAM,OAAO,aAAa;AAAA,MACzC,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,IAAI;AAAA,IACb,CAAC;AAED,WAAO,aAAa,eAAe,OAAO;AAAA,EAC5C,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAkBA,eAAsB,UACpB,MACA,QACwB;AACxB,MAAI;AACF,UAAM,SAAS,aAAa,MAAM;AAClC,UAAM,QAAQ,KAAK,QAAQ,QAAQ,EAAE;AACrC,UAAM,YAAY,UAAU,QAAQ,KAAK,CAAC;AAE1C,UAAM,SAAS,MAAM,OAAO,aAAa;AAAA,MACvC,SAAS;AAAA,MACT,KAAK;AAAA,QACH;AAAA,UACE,MAAM;AAAA,UACN,MAAM;AAAA,UACN,QAAQ,CAAC,EAAE,MAAM,MAAM,MAAM,UAAU,CAAC;AAAA,UACxC,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,UACvC,iBAAiB;AAAA,QACnB;AAAA,MACF;AAAA,MACA,cAAc;AAAA,MACd,MAAM,CAAC,OAAO,SAAS,CAAC;AAAA,IAC1B,CAAC;AAED,WAAO,OAAO,MAAM;AAAA,EACtB,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAoGA,eAAsB,aACpB,MACA,QACA,SACwB;AACxB,QAAM,MAAM,MAAM,cAAc,MAAM,UAAU,MAAM;AACtD,MAAI,CAAC,IAAK,QAAO,SAAS,YAAY;AAEtC,MAAI;AAEF,QAAI,IAAI,WAAW,SAAS,KAAK,IAAI,WAAW,UAAU,GAAG;AAC3D,aAAO;AAAA,IACT;AAGA,QAAI,IAAI,WAAW,SAAS,GAAG;AAC7B,YAAM,OAAO,IAAI,MAAM,CAAC;AACxB,aAAO,wBAAwB,IAAI;AAAA,IACrC;AAGA,QAAI,qDAAqD,KAAK,GAAG,GAAG;AAClE,aAAO,wBAAwB,GAAG;AAAA,IACpC;AAGA,QAAI,IAAI,WAAW,OAAO,GAAG;AAC3B,YAAM,OAAO,IAAI,MAAM,CAAC;AACxB,aAAO,uBAAuB,IAAI;AAAA,IACpC;AAIA,UAAM,WAAW,IAAI;AAAA,MACnB;AAAA,IACF;AACA,QAAI,UAAU;AACZ,YAAM,SAAS,aAAa,MAAM;AAClC,YAAM,WAAW,KAAK,SAAS,CAAC,CAAC;AACjC,YAAM,UAAU,OAAO,SAAS,CAAC,CAAC;AAElC,YAAM,cAAc;AAAA,QAClB;AAAA,UACE,MAAM,SAAS,CAAC,MAAM,WAAW,aAAa;AAAA,UAC9C,MAAM;AAAA,UACN,QAAQ,CAAC,EAAE,MAAM,WAAW,MAAM,UAAU,CAAC;AAAA,UAC7C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,SAAS,CAAC;AAAA,UACtC,iBAAiB;AAAA,QACnB;AAAA,MACF;AAEA,UAAI,WAAY,MAAM,OAAO,aAAa;AAAA,QACxC,SAAS;AAAA,QACT,KAAK;AAAA,QACL,cAAc,SAAS,CAAC,MAAM,WAAW,aAAa;AAAA,QACtD,MAAM,CAAC,OAAO;AAAA,MAChB,CAAC;AAGD,UAAI,SAAS,WAAW,SAAS,GAAG;AAClC,mBAAW,wBAAwB,SAAS,MAAM,CAAC,CAAC;AAAA,MACtD;AAGA,YAAM,MAAM,MAAM,MAAM,QAAQ;AAChC,YAAM,WAAW,MAAM,IAAI,KAAK;AAChC,UAAI,QAAQ,SAAS,SAAS,SAAS,aAAa;AAEpD,UAAI,SAAS,MAAM,WAAW,SAAS,GAAG;AACxC,gBAAQ,wBAAwB,MAAM,MAAM,CAAC,CAAC;AAAA,MAChD;AAEA,aAAO,SAAS,SAAS,YAAY;AAAA,IACvC;AAGA,QAAI,IAAI,WAAW,OAAO,GAAG;AAC3B,aAAO;AAAA,IACT;AAEA,WAAO,SAAS,YAAY;AAAA,EAC9B,SAAS,OAAO;AACd,YAAQ,KAAK,6BAA6B,KAAK;AAC/C,WAAO,SAAS,YAAY;AAAA,EAC9B;AACF;;;AGreA,SAAS,UAAU,WAAW,mBAAmB;;;ACFjD,SAAS,YAAAC,iBAAgB;;;ACalB,SAAS,gBAAgB,OAG9B;AACA,MAAI,CAAC,SAAS,MAAM,KAAK,EAAE,WAAW,GAAG;AACvC,WAAO,EAAE,OAAO,OAAO,OAAO,uBAAuB;AAAA,EACvD;AAEA,QAAM,UAAU,MAAM,KAAK,EAAE,YAAY;AAGzC,MAAI,QAAQ,WAAW,IAAI,GAAG;AAC5B,WAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAGA,MAAI,CAAC,0CAA0C,KAAK,OAAO,GAAG;AAC5D,WAAO;AAAA,MACL,OAAO;AAAA,MACP,OACE;AAAA,IACJ;AAAA,EACF;AAGA,MAAI,QAAQ,SAAS,IAAI,GAAG;AAC1B,WAAO,EAAE,OAAO,OAAO,OAAO,0CAA0C;AAAA,EAC1E;AAGA,MAAI,QAAQ,SAAS,IAAI;AACvB,WAAO,EAAE,OAAO,OAAO,OAAO,qCAAqC;AAAA,EACrE;AAEA,SAAO,EAAE,OAAO,KAAK;AACvB;AAMO,SAAS,kBAAkB,KAIhC;AACA,MAAI,CAAC,KAAK;AACR,WAAO,EAAE,OAAO,OAAO,OAAO,yBAAyB;AAAA,EACzD;AAGA,MAAI,IAAI,WAAW,OAAO,GAAG;AAC3B,UAAM,YAAY,IAAI,KAAK,CAAC,GAAG,CAAC,EAAE;AAElC,QAAI,YAAY,kBAAkB;AAChC,aAAO;AAAA,QACL,OAAO;AAAA,QACP,OAAO,iBAAiB,YAAY,MAAM,QAAQ,CAAC,CAAC,eAAe,mBAAmB,IAAI;AAAA,QAC1F;AAAA,MACF;AAAA,IACF;AAEA,WAAO,EAAE,OAAO,MAAM,UAAU;AAAA,EAClC;AAGA,QAAM,eAAe,CAAC,WAAW,YAAY,WAAW,WAAW,OAAO;AAC1E,QAAM,iBAAiB,aAAa,KAAK,CAAC,WAAW,IAAI,WAAW,MAAM,CAAC;AAE3E,MAAI,CAAC,gBAAgB;AAEnB,QAAI,qDAAqD,KAAK,GAAG,GAAG;AAClE,aAAO,EAAE,OAAO,KAAK;AAAA,IACvB;AAEA,WAAO;AAAA,MACL,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,EACF;AAEA,SAAO,EAAE,OAAO,KAAK;AACvB;AAmBA,eAAsB,mBACpB,KACA,SAC8C;AAC9C,QAAM,WAAW,SAAS,YAAY;AACtC,QAAM,UAAU,SAAS,WAAW;AAGpC,QAAM,mBAAmB,kBAAkB,GAAG;AAC9C,MAAI,CAAC,iBAAiB,OAAO;AAC3B,UAAM,IAAI,MAAM,iBAAiB,KAAK;AAAA,EACxC;AAGA,MAAI,IAAI,WAAW,OAAO,GAAG;AAC3B,WAAO,EAAE,OAAO,MAAM,WAAW,iBAAiB,UAAU;AAAA,EAC9D;AAGA,MAAI,IAAI,WAAW,SAAS,GAAG;AAC7B,WAAO,EAAE,OAAO,KAAK;AAAA,EACvB;AAGA,MAAI,WAAW;AAGf,MAAI,IAAI,WAAW,SAAS,GAAG;AAC7B,eAAW,wBAAwB,IAAI,MAAM,CAAC,CAAC;AAAA,EACjD,WAAW,qDAAqD,KAAK,GAAG,GAAG;AACzE,eAAW,wBAAwB,GAAG;AAAA,EACxC,WAAW,IAAI,WAAW,OAAO,GAAG;AAClC,eAAW,uBAAuB,IAAI,MAAM,CAAC,CAAC;AAAA,EAChD;AAEA,MAAI;AACF,UAAM,aAAa,IAAI,gBAAgB;AACvC,UAAM,YAAY,WAAW,MAAM,WAAW,MAAM,GAAG,OAAO;AAE9D,UAAM,WAAW,MAAM,MAAM,UAAU;AAAA,MACrC,QAAQ;AAAA,MACR,QAAQ,WAAW;AAAA,IACrB,CAAC;AAED,iBAAa,SAAS;AAEtB,UAAM,gBAAgB,SAAS,QAAQ,IAAI,gBAAgB;AAC3D,QAAI,eAAe;AACjB,YAAM,YAAY,SAAS,eAAe,EAAE;AAE5C,UAAI,YAAY,UAAU;AACxB,cAAM,IAAI;AAAA,UACR,sBAAsB,YAAY,MAAM,QAAQ,CAAC,CAAC,eAAe,WAAW,IAAI;AAAA,QAClF;AAAA,MACF;AAEA,aAAO,EAAE,OAAO,MAAM,UAAU;AAAA,IAClC;AAGA,YAAQ;AAAA,MACN,uCAAuC,GAAG,uBAAuB,WAAW,IAAI;AAAA,IAClF;AACA,WAAO,EAAE,OAAO,KAAK;AAAA,EACvB,SAAS,OAAY;AACnB,QAAI,MAAM,SAAS,cAAc;AAC/B,cAAQ,KAAK,mCAAmC,GAAG,EAAE;AACrD,aAAO,EAAE,OAAO,KAAK;AAAA,IACvB;AAGA,QAAI,MAAM,QAAQ,SAAS,SAAS,GAAG;AACrC,YAAM;AAAA,IACR;AAGA,YAAQ,KAAK,gCAAgC,MAAM,OAAO,EAAE;AAC5D,WAAO,EAAE,OAAO,KAAK;AAAA,EACvB;AACF;;;ADrLO,IAAM,oBAAoB;AAAA,EAC/B;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,SAAS,MAAM,SAAS,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,OAAO,CAAC;AAAA,IACpC,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ;AAAA,MACN,EAAE,MAAM,SAAS,MAAM,SAAS;AAAA,MAChC,EAAE,MAAM,YAAY,MAAM,UAAU;AAAA,IACtC;AAAA,IACA,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,IACvC,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ,CAAC,EAAE,MAAM,QAAQ,MAAM,UAAU,CAAC;AAAA,IAC1C,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,OAAO,CAAC;AAAA,IACpC,iBAAiB;AAAA,EACnB;AACF;AAEO,IAAM,qBAAqB;AAAA,EAChC;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ;AAAA,MACN,EAAE,MAAM,SAAS,MAAM,SAAS;AAAA,MAChC,EAAE,MAAM,SAAS,MAAM,UAAU;AAAA,MACjC,EAAE,MAAM,YAAY,MAAM,UAAU;AAAA,IACtC;AAAA,IACA,SAAS,CAAC,EAAE,MAAM,IAAI,MAAM,UAAU,CAAC;AAAA,IACvC,iBAAiB;AAAA,EACnB;AACF;AAEO,IAAM,mBAAmB;AAAA,EAC9B;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ;AAAA,MACN,EAAE,MAAM,QAAQ,MAAM,UAAU;AAAA,MAChC,EAAE,MAAM,OAAO,MAAM,SAAS;AAAA,MAC9B,EAAE,MAAM,SAAS,MAAM,SAAS;AAAA,IAClC;AAAA,IACA,SAAS,CAAC;AAAA,IACV,iBAAiB;AAAA,EACnB;AAAA,EACA;AAAA,IACE,MAAM;AAAA,IACN,MAAM;AAAA,IACN,QAAQ;AAAA,MACN,EAAE,MAAM,QAAQ,MAAM,UAAU;AAAA,MAChC,EAAE,MAAM,KAAK,MAAM,UAAU;AAAA,IAC/B;AAAA,IACA,SAAS,CAAC;AAAA,IACV,iBAAiB;AAAA,EACnB;AACF;AAeA,IAAM,WAAW,OAAO,MAAM,KAAK,KAAK,EAAE;AAW1C,eAAsB,oBACpB,OACA,aACA,gBAAgB,GAChB,QAC2B;AAE3B,QAAM,aAAa,gBAAgB,KAAK;AACxC,MAAI,CAAC,WAAW,OAAO;AACrB,UAAM,IAAI,MAAM,WAAW,KAAK;AAAA,EAClC;AAEA,QAAM,SAAS,aAAa,MAAM;AAClC,QAAM,WAAW,WAAW,OAAO,aAAa;AAEhD,QAAM,CAAC,WAAW,OAAO,YAAY,OAAO,IAAI,MAAM,QAAQ,IAAI;AAAA,IAChE,OAAO,aAAa;AAAA,MAClB,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,KAAK;AAAA,IACd,CAAC;AAAA,IACD,OAAO,aAAa;AAAA,MAClB,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,OAAO,QAAQ;AAAA,IACxB,CAAC;AAAA,IACD,OAAO,aAAa;AAAA,MAClB,SAAS;AAAA,MACT,KAAK;AAAA,MACL,cAAc;AAAA,MACd,MAAM,CAAC,WAA4B;AAAA,IACrC,CAAC;AAAA,IACD,OAAO,WAAW,EAAE,SAAS,YAA6B,CAAC;AAAA,EAC7D,CAAC;AAED,QAAM,YAAY,WAAW,MAAM,OAAO,OAAO,CAAC;AAElD,SAAO,EAAE,WAAW,OAAO,WAAW,YAAY,UAAU,MAAM;AACpE;AAYA,eAAsB,cACpB,OACA,cACA,gBAAgB,GAChB,QACA;AAEA,QAAM,aAAa,gBAAgB,KAAK;AACxC,MAAI,CAAC,WAAW,OAAO;AACrB,UAAM,IAAI,MAAM,WAAW,KAAK;AAAA,EAClC;AAEA,QAAM,OAAO,MAAM;AAAA,IACjB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AAEA,MAAI,CAAC,KAAK,WAAW;AACnB,UAAM,IAAI,MAAM,GAAG,KAAK,uBAAuB;AAAA,EACjD;AAEA,SAAO;AAAA,IACL,SAAS;AAAA,IACT,KAAK;AAAA,IACL,cAAc;AAAA,IACd,MAAM,CAAC,OAAO,cAA+B,KAAK,QAAQ;AAAA,IAC1D,OAAO,KAAK,YAAY,KAAK,KAAK;AAAA,IAClC,KAAK;AAAA,IACL,OAAO;AAAA,MACL;AAAA,MACA,WAAW,KAAK;AAAA,MAChB,OAAO,KAAK;AAAA,MACZ,UAAU,KAAK;AAAA,IACjB;AAAA,EACF;AACF;AAeO,SAAS,aAAa,MAAc,KAAa,OAAe;AAErE,MAAI,QAAQ,YAAY,MAAM,WAAW,OAAO,GAAG;AACjD,UAAM,aAAa,kBAAkB,KAAK;AAC1C,QAAI,CAAC,WAAW,OAAO;AACrB,YAAM,IAAI,MAAM,WAAW,KAAK;AAAA,IAClC;AAAA,EACF;AAGA,MAAI,QAAQ,YAAY,CAAC,MAAM,WAAW,OAAO,GAAG;AAClD,YAAQ;AAAA,MACN;AAAA,IACF;AAAA,EACF;AAEA,QAAM,OAAOC;AAAA,IACX,KAAK,SAAS,MAAM,IAAI,OAAO,GAAG,IAAI;AAAA,EACxC;AAEA,SAAO;AAAA,IACL,SAAS;AAAA,IACT,KAAK;AAAA,IACL,cAAc;AAAA,IACd,MAAM,CAAC,MAAM,KAAK,KAAK;AAAA,EACzB;AACF;AAKO,SAAS,aAAa,MAAc,MAAc;AACvD,QAAM,OAAOA;AAAA,IACX,KAAK,SAAS,MAAM,IAAI,OAAO,GAAG,IAAI;AAAA,EACxC;AAEA,SAAO;AAAA,IACL,SAAS;AAAA,IACT,KAAK;AAAA,IACL,cAAc;AAAA,IACd,MAAM,CAAC,MAAM,IAAqB;AAAA,EACpC;AACF;AAEO,IAAM,mBAAmB;AAAA,EAC9B,QAAQ;AAAA,EACR,KAAK;AAAA,EACL,aAAa;AAAA,EACb,OAAO;AAAA,EACP,SAAS;AAAA,EACT,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,UAAU;AACZ;;;AD7NO,SAAS,oBACd,OACA,aACA,gBAAgB,GAChB,QACA;AACA,QAAM,CAAC,MAAM,OAAO,IAAI,SAAkC,IAAI;AAC9D,QAAM,CAAC,SAAS,UAAU,IAAI,SAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAI,SAAwB,IAAI;AAEtD,YAAU,MAAM;AACd,QAAI,CAAC,SAAS,CAAC,eAAe,MAAM,KAAK,EAAE,SAAS,GAAG;AACrD,cAAQ,IAAI;AACZ,eAAS,IAAI;AACb,iBAAW,KAAK;AAChB;AAAA,IACF;AAGA,UAAM,aAAa,gBAAgB,KAAK;AACxC,QAAI,CAAC,WAAW,OAAO;AACrB,cAAQ,IAAI;AACZ,eAAS,WAAW,SAAS,cAAc;AAC3C,iBAAW,KAAK;AAChB;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,UAAM,QAAQ,WAAW,MAAM;AAC7B,0BAAoB,OAAO,aAAa,eAAe,MAAM,EAC1D,KAAK,CAAC,MAAM;AACX,YAAI,CAAC,UAAW,SAAQ,CAAC;AAAA,MAC3B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,YAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,MACpC,CAAC,EACA,QAAQ,MAAM;AACb,YAAI,CAAC,UAAW,YAAW,KAAK;AAAA,MAClC,CAAC;AAAA,IACL,GAAG,GAAG;AAEN,WAAO,MAAM;AACX,kBAAY;AACZ,mBAAa,KAAK;AAAA,IACpB;AAAA,EACF,GAAG,CAAC,OAAO,aAAa,eAAe,MAAM,CAAC;AAE9C,SAAO,EAAE,MAAM,SAAS,MAAM;AAChC;AAoBO,SAAS,eAAe,QAA0B;AACvD,QAAM,CAAC,IAAI,KAAK,IAAI,SAA4B,IAAI;AACpD,QAAM,CAAC,SAAS,UAAU,IAAI,SAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAI,SAAwB,IAAI;AAEtD,QAAM,UAAU;AAAA,IACd,OAAO,OAAe,cAAsB,gBAAgB,MAAM;AAChE,iBAAW,IAAI;AACf,eAAS,IAAI;AACb,YAAM,IAAI;AAEV,UAAI;AACF,cAAM,WAAW,MAAM;AAAA,UACrB;AAAA,UACA;AAAA,UACA;AAAA,UACA;AAAA,QACF;AACA,cAAM,QAAQ;AACd,eAAO;AAAA,MACT,SAAS,GAAQ;AACf,iBAAS,EAAE,OAAO;AAClB,eAAO;AAAA,MACT,UAAE;AACA,mBAAW,KAAK;AAAA,MAClB;AAAA,IACF;AAAA,IACA,CAAC;AAAA,EACH;AAEA,SAAO,EAAE,SAAS,IAAI,SAAS,MAAM;AACvC;AAaO,SAAS,oBAAoB;AAClC,QAAM,gBAAgB;AAAA,IACpB,CAAC,MAAc,KAAa,UAAkB;AAC5C,aAAO,aAAa,MAAM,KAAK,KAAK;AAAA,IACtC;AAAA,IACA,CAAC;AAAA,EACH;AAEA,QAAM,YAAY,YAAY,CAAC,MAAc,QAAgB;AAC3D,WAAO,aAAa,MAAM,UAAU,GAAG;AAAA,EACzC,GAAG,CAAC,CAAC;AAEL,QAAM,qBAAqB,YAAY,OAAO,MAAc,QAAgB;AAC1E,UAAM,mBAAmB,GAAG;AAC5B,WAAO,aAAa,MAAM,UAAU,GAAG;AAAA,EACzC,GAAG,CAAC,CAAC;AAEL,QAAM,SAAS,YAAY,CAAC,MAAc,QAAgB;AACxD,WAAO,aAAa,MAAM,OAAO,GAAG;AAAA,EACtC,GAAG,CAAC,CAAC;AAEL,QAAM,aAAa,YAAY,CAAC,MAAc,WAAmB;AAC/D,WAAO,aAAa,MAAM,eAAe,MAAM;AAAA,EACjD,GAAG,CAAC,CAAC;AAEL,QAAM,YAAY,YAAY,CAAC,MAAc,WAAmB;AAC9D,WAAO,aAAa,MAAM,cAAc,MAAM;AAAA,EAChD,GAAG,CAAC,CAAC;AAEL,QAAM,aAAa,YAAY,CAAC,MAAc,WAAmB;AAC/D,WAAO,aAAa,MAAM,eAAe,MAAM;AAAA,EACjD,GAAG,CAAC,CAAC;AAEL,QAAM,iBAAiB,YAAY,CAAC,MAAc,SAAiB;AACjE,WAAO,aAAa,MAAM,eAAe,IAAI;AAAA,EAC/C,GAAG,CAAC,CAAC;AAEL,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,gBAAgB;AAAA,EAClB;AACF;AAaO,SAAS,aAAa;AAC3B,QAAM,UAAU,YAAY,CAAC,MAAc,SAAiB;AAC1D,WAAO,aAAa,MAAM,IAAI;AAAA,EAChC,GAAG,CAAC,CAAC;AAEL,SAAO,EAAE,QAAQ;AACnB;;;AJnMO,SAAS,WACd,SACA,QACA;AACA,QAAM,CAAC,MAAM,OAAO,IAAIC,UAAwB,IAAI;AACpD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,WAAW,CAAC,QAAQ,WAAW,IAAI,KAAK,QAAQ,WAAW,IAAI;AAClE,cAAQ,IAAI;AACZ;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,kBAAc,SAAS,MAAM,EAC1B,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,SAAQ,CAAC;AAAA,IAC3B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,OAAO,CAAC;AAEZ,SAAO,EAAE,MAAM,SAAS,MAAM;AAChC;AAaO,SAAS,cACd,MACA,QACA;AACA,QAAM,CAAC,SAAS,UAAU,IAAID,UAAwB,IAAI;AAC1D,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,iBAAW,IAAI;AACf;AAAA,IACF;AACA,UAAM,WAAW,KAAK,SAAS,MAAM,IAAI,OAAO,GAAG,IAAI;AAEvD,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,gBAAY,UAAU,MAAM,EACzB,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,YAAW,CAAC;AAAA,IAC9B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,IAAI,CAAC;AAET,SAAO,EAAE,SAAS,SAAS,MAAM;AACnC;AAiBO,SAAS,cACd,SACA,QACA;AACA,QAAM,EAAE,MAAM,SAAS,MAAM,IAAI,WAAW,SAAS,MAAM;AAC3D,QAAM,YAAY,UACd,GAAG,QAAQ,MAAM,GAAG,CAAC,CAAC,MAAM,QAAQ,MAAM,EAAE,CAAC,KAC7C;AAEJ,SAAO;AAAA,IACL,aAAa,QAAQ;AAAA,IACrB,SAAS;AAAA,IACT;AAAA,IACA;AAAA,EACF;AACF;AAmBO,SAAS,cAAc,QAA0B;AACtD,QAAM,CAAC,SAAS,UAAU,IAAID,UAAwB,IAAI;AAC1D,QAAM,CAAC,MAAM,OAAO,IAAIA,UAAwB,IAAI;AACpD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,QAAM,UAAUE,aAAY,OAAO,UAAkB;AACnD,UAAM,UAAU,MAAM,KAAK;AAC3B,aAAS,IAAI;AAEb,QAAI,CAAC,SAAS;AACZ,iBAAW,IAAI;AACf,cAAQ,IAAI;AACZ;AAAA,IACF;AAEA,QAAI,QAAQ,WAAW,IAAI,KAAK,QAAQ,WAAW,IAAI;AACrD,iBAAW,OAAO;AAClB,iBAAW,IAAI;AACf,YAAM,IAAI,MAAM,cAAc,SAAS,MAAM;AAC7C,cAAQ,CAAC;AACT,iBAAW,KAAK;AAChB;AAAA,IACF;AAEA,UAAM,WAAW,QAAQ,SAAS,MAAM,IAAI,UAAU,GAAG,OAAO;AAChE,YAAQ,QAAQ;AAChB,eAAW,IAAI;AAEf,QAAI;AACF,YAAM,OAAO,MAAM,YAAY,UAAU,MAAM;AAC/C,iBAAW,IAAI;AACf,UAAI,CAAC,KAAM,UAAS,GAAG,QAAQ,YAAY;AAAA,IAC7C,SAAS,GAAQ;AACf,eAAS,EAAE,OAAO;AAClB,iBAAW,IAAI;AAAA,IACjB,UAAE;AACA,iBAAW,KAAK;AAAA,IAClB;AAAA,EACF,GAAG,CAAC,CAAC;AAEL,SAAO,EAAE,SAAS,SAAS,MAAM,SAAS,MAAM;AAClD;AAaO,SAAS,WACd,MACA,KACA,QACA;AACA,QAAM,CAAC,OAAO,QAAQ,IAAIF,UAAwB,IAAI;AACtD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,eAAS,IAAI;AACb;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,kBAAc,MAAM,KAAK,MAAM,EAC5B,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,UAAS,CAAC;AAAA,IAC5B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,MAAM,GAAG,CAAC;AAEd,SAAO,EAAE,OAAO,SAAS,MAAM;AACjC;AAcO,SAAS,aACd,MACA,QACA;AACA,QAAM,CAAC,KAAK,MAAM,IAAID,UAAwB,IAAI;AAClD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,aAAO,IAAI;AACX;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,iBAAa,MAAM,MAAM,EACtB,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,QAAO,CAAC;AAAA,IAC1B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,IAAI,CAAC;AAET,SAAO,EAAE,KAAK,SAAS,MAAM;AAC/B;AAcO,SAAS,YACd,MACA,QACA;AACA,QAAM,CAAC,OAAO,QAAQ,IAAID,UAAwB,IAAI;AACtD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,eAAS,IAAI;AACb;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,aAAS,MAAM,MAAM,EAClB,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,UAAS,CAAC;AAAA,IAC5B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,IAAI,CAAC;AAET,SAAO,EAAE,OAAO,SAAS,MAAM;AACjC;AAaO,SAAS,eACd,MACA,QACA;AACA,QAAM,CAAC,UAAU,WAAW,IAAID,UAAwB,IAAI;AAC5D,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,kBAAY,IAAI;AAChB;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,gBAAY,MAAM,MAAM,EACrB,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,aAAY,CAAC;AAAA,IAC/B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,IAAI,CAAC;AAET,SAAO,EAAE,UAAU,SAAS,MAAM;AACpC;AAeO,SAAS,aACd,MACA,QACA;AACA,QAAM,CAAC,QAAQ,SAAS,IAAID,UAAwB,IAAI;AACxD,QAAM,CAAC,SAAS,UAAU,IAAIA,UAAS,KAAK;AAC5C,QAAM,CAAC,OAAO,QAAQ,IAAIA,UAAwB,IAAI;AAEtD,EAAAC,WAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT,gBAAU,IAAI;AACd;AAAA,IACF;AAEA,QAAI,YAAY;AAChB,eAAW,IAAI;AACf,aAAS,IAAI;AAEb,cAAU,MAAM,MAAM,EACnB,KAAK,CAAC,MAAM;AACX,UAAI,CAAC,UAAW,WAAU,CAAC;AAAA,IAC7B,CAAC,EACA,MAAM,CAAC,MAAM;AACZ,UAAI,CAAC,UAAW,UAAS,EAAE,OAAO;AAAA,IACpC,CAAC,EACA,QAAQ,MAAM;AACb,UAAI,CAAC,UAAW,YAAW,KAAK;AAAA,IAClC,CAAC;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,IAAI,CAAC;AAET,QAAM,aAAa,SAAS,IAAI,KAAK,SAAS,GAAI,IAAI;AACtD,QAAM,kBAAkB,SACpB,KAAK,OAAO,SAAS,MAAO,KAAK,IAAI,MAAM,MAAO,KAAK,KAAK,GAAG,IAC/D;AACJ,QAAM,YAAY,SAAS,SAAS,MAAO,KAAK,IAAI,IAAI;AAExD,SAAO,EAAE,QAAQ,YAAY,iBAAiB,WAAW,SAAS,MAAM;AAC1E;","names":["useState","useEffect","useCallback","namehash","namehash","useState","useEffect","useCallback"]}
|