@githat/nextjs 0.4.0 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/provider.tsx","../src/config.ts","../src/client.ts","../src/hooks.ts","../src/data.ts","../src/components/SignInForm.tsx","../src/components/SignUpForm.tsx","../src/components/SignInButton.tsx","../src/components/SignUpButton.tsx","../src/components/UserButton.tsx","../src/components/OrgSwitcher.tsx","../src/components/VerifiedBadge.tsx","../src/components/ProtectedRoute.tsx"],"sourcesContent":["'use client';\n\nimport React, { createContext, useState, useEffect, useCallback, useMemo, useRef } from 'react';\nimport type { GitHatConfig, GitHatContextValue, GitHatUser, GitHatOrg, SignUpData, SignUpResult } from './types';\nimport { TOKEN_KEYS, resolveConfig } from './config';\nimport { createClient } from './client';\n\nexport const GitHatContext = createContext<GitHatContextValue | null>(null);\n\ninterface GitHatProviderProps {\n config: GitHatConfig;\n children: React.ReactNode;\n}\n\nexport function GitHatProvider({ config: rawConfig, children }: GitHatProviderProps) {\n const config = useMemo(() => resolveConfig(rawConfig), [rawConfig]);\n const useCookies = config.tokenStorage === 'cookie';\n\n // Create client with cookie mode awareness\n const clientRef = useRef(createClient(config.apiUrl, config.publishableKey, { useCookies }));\n\n const [user, setUser] = useState<GitHatUser | null>(null);\n const [org, setOrg] = useState<GitHatOrg | null>(null);\n const [isSignedIn, setIsSignedIn] = useState(false);\n const [isLoading, setIsLoading] = useState(true);\n const [authError, setAuthError] = useState<string | null>(null);\n\n // Validate stored token/session on mount\n useEffect(() => {\n const validateSession = async () => {\n try {\n // In cookie mode, we always try to validate (cookies sent automatically)\n // In localStorage mode, only validate if we have a token\n if (!useCookies) {\n const token = localStorage.getItem(TOKEN_KEYS.accessToken);\n const storedUser = localStorage.getItem(TOKEN_KEYS.user);\n if (!token || !storedUser) {\n setIsLoading(false);\n return;\n }\n }\n\n const data = await clientRef.current.fetchApi<{ user: GitHatUser; currentOrg: GitHatOrg }>('/auth/me');\n\n if (data.user) {\n setUser(data.user);\n setOrg(data.currentOrg || null);\n setIsSignedIn(true);\n setAuthError(null);\n\n // Store user info in localStorage for client-side access (even in cookie mode)\n if (!useCookies) {\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.currentOrg) {\n localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.currentOrg));\n }\n }\n }\n } catch (err: unknown) {\n const error = err as Error;\n if (error.message === 'Session expired') {\n clientRef.current.clearAuth();\n } else if (!useCookies) {\n // Network error in localStorage mode — try to use stored user\n const storedUser = localStorage.getItem(TOKEN_KEYS.user);\n if (storedUser) {\n try {\n setUser(JSON.parse(storedUser));\n setIsSignedIn(true);\n } catch {}\n }\n setAuthError(error.message || 'Failed to verify session');\n }\n // In cookie mode with error, just stay logged out\n }\n setIsLoading(false);\n };\n\n validateSession();\n }, [useCookies]);\n\n // Listen for auth-changed events\n useEffect(() => {\n const handleAuthChanged = (e: Event) => {\n const detail = (e as CustomEvent).detail;\n if (detail?.signedIn === false) {\n setIsSignedIn(false);\n setUser(null);\n setOrg(null);\n } else if (detail?.signedIn === true && detail?.user) {\n setUser(detail.user);\n setIsSignedIn(true);\n if (detail.org) setOrg(detail.org);\n }\n };\n window.addEventListener('githat:auth-changed', handleAuthChanged);\n return () => window.removeEventListener('githat:auth-changed', handleAuthChanged);\n }, []);\n\n const signIn = useCallback(async (email: string, password: string) => {\n // Build login URL with setCookie param if in cookie mode\n const loginUrl = useCookies ? '/auth/login?setCookie=true' : '/auth/login';\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n user: GitHatUser;\n org: GitHatOrg;\n }>(loginUrl, {\n method: 'POST',\n body: JSON.stringify({ email, password }),\n });\n\n // In localStorage mode, store tokens\n if (!useCookies && data.accessToken && data.refreshToken) {\n localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setUser(data.user);\n setOrg(data.org || null);\n setIsSignedIn(true);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user: data.user, org: data.org, signedIn: true },\n }));\n }, [useCookies]);\n\n const signUp = useCallback(async (signUpData: SignUpData): Promise<SignUpResult> => {\n // Build register URL with setCookie param if in cookie mode\n const registerUrl = useCookies ? '/auth/register?setCookie=true' : '/auth/register';\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n user: GitHatUser;\n org: GitHatOrg;\n }>(registerUrl, {\n method: 'POST',\n body: JSON.stringify(signUpData),\n });\n\n // In localStorage mode, store tokens\n if (!useCookies && data.accessToken && data.refreshToken) {\n localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setUser(data.user);\n setOrg(data.org || null);\n setIsSignedIn(true);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user: data.user, org: data.org, signedIn: true },\n }));\n\n return { requiresVerification: !data.user.emailVerified, email: signUpData.email };\n }, [useCookies]);\n\n const signOut = useCallback(async () => {\n try {\n // In cookie mode, logout endpoint clears cookies\n const logoutUrl = useCookies ? '/auth/logout?setCookie=true' : '/auth/logout';\n await clientRef.current.fetchApi(logoutUrl, { method: 'POST' });\n } catch {}\n clientRef.current.clearAuth();\n setIsSignedIn(false);\n setUser(null);\n setOrg(null);\n if (typeof window !== 'undefined' && config.afterSignOutUrl) {\n window.location.href = config.afterSignOutUrl;\n }\n }, [config.afterSignOutUrl, useCookies]);\n\n const switchOrg = useCallback(async (orgId: string) => {\n try {\n const switchUrl = useCookies\n ? `/user/orgs/${orgId}/switch?setCookie=true`\n : `/user/orgs/${orgId}/switch`;\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n org: GitHatOrg;\n }>(switchUrl, { method: 'POST' });\n\n // In localStorage mode, store new tokens\n if (!useCookies) {\n if (data.accessToken) localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n if (data.refreshToken) localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setOrg(data.org);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user, org: data.org, signedIn: true },\n }));\n } catch (e) {\n console.error('Org switch failed:', e);\n }\n }, [user, useCookies]);\n\n const value = useMemo<GitHatContextValue>(() => ({\n user, org, isSignedIn, isLoading, authError, config,\n signIn, signUp, signOut, switchOrg,\n }), [user, org, isSignedIn, isLoading, authError, config, signIn, signUp, signOut, switchOrg]);\n\n return (\n <GitHatContext.Provider value={value}>\n {children}\n </GitHatContext.Provider>\n );\n}\n","import type { GitHatConfig } from './types';\n\nexport const DEFAULT_API_URL = 'https://api.githat.io';\n\nexport const TOKEN_KEYS = {\n accessToken: 'githat_access_token',\n refreshToken: 'githat_refresh_token',\n user: 'githat_user',\n org: 'githat_org',\n} as const;\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\nexport function resolveConfig(config: GitHatConfig): Required<GitHatConfig> {\n return {\n publishableKey: config.publishableKey,\n apiUrl: config.apiUrl || DEFAULT_API_URL,\n signInUrl: config.signInUrl || '/sign-in',\n signUpUrl: config.signUpUrl || '/sign-up',\n afterSignInUrl: config.afterSignInUrl || '/dashboard',\n afterSignOutUrl: config.afterSignOutUrl || '/',\n tokenStorage: config.tokenStorage || 'localStorage',\n };\n}\n","import { TOKEN_KEYS } from './config';\n\ninterface ClientOptions {\n /**\n * When true, use credentials: 'include' for all requests\n * and don't send Authorization header (cookies handle auth)\n */\n useCookies?: boolean;\n}\n\nlet _refreshPromise: Promise<boolean> | null = null;\n\nasync function refreshTokens(\n apiUrl: string,\n appKey: string,\n useCookies: boolean\n): Promise<boolean> {\n // In cookie mode, refresh token comes from cookie\n // In localStorage mode, we need to send it in the body\n const refreshToken =\n typeof window !== 'undefined' && !useCookies\n ? localStorage.getItem(TOKEN_KEYS.refreshToken)\n : null;\n\n // In localStorage mode, we need a refresh token\n if (!useCookies && !refreshToken) return false;\n\n let orgId: string | null = null;\n try {\n const orgStr = localStorage.getItem(TOKEN_KEYS.org);\n if (orgStr) orgId = JSON.parse(orgStr).id;\n } catch {}\n\n try {\n const refreshUrl = useCookies\n ? `${apiUrl}/auth/refresh?setCookie=true`\n : `${apiUrl}/auth/refresh`;\n\n const res = await fetch(refreshUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-GitHat-App-Key': appKey,\n },\n credentials: useCookies ? 'include' : 'same-origin',\n body: JSON.stringify(useCookies ? { orgId } : { refreshToken, orgId }),\n });\n\n if (!res.ok) return false;\n\n const data = await res.json();\n\n // In localStorage mode, store the new tokens\n if (!useCookies) {\n if (data.accessToken) localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n if (data.refreshToken) localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n }\n\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n return true;\n } catch {\n return false;\n }\n}\n\nfunction clearAuth() {\n if (typeof window === 'undefined') return;\n Object.values(TOKEN_KEYS).forEach((key) => localStorage.removeItem(key));\n window.dispatchEvent(\n new CustomEvent('githat:auth-changed', {\n detail: { user: null, org: null, signedIn: false },\n })\n );\n}\n\nexport function createClient(\n apiUrl: string,\n appKey: string,\n options: ClientOptions = {}\n) {\n const { useCookies = false } = options;\n\n async function fetchApi<T = unknown>(\n endpoint: string,\n fetchOptions: RequestInit = {}\n ): Promise<T> {\n const url = `${apiUrl}${endpoint}`;\n\n // In localStorage mode, get token from storage\n // In cookie mode, cookies are sent automatically with credentials: 'include'\n const token =\n typeof window !== 'undefined' && !useCookies\n ? localStorage.getItem(TOKEN_KEYS.accessToken)\n : null;\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'X-GitHat-App-Key': appKey,\n ...(token && { Authorization: `Bearer ${token}` }),\n ...(fetchOptions.headers as Record<string, string>),\n };\n\n let response: Response;\n try {\n response = await fetch(url, {\n ...fetchOptions,\n headers,\n credentials: useCookies ? 'include' : 'same-origin',\n });\n } catch (networkError: unknown) {\n // Network errors (CORS blocked, offline, DNS failure) throw TypeError\n if (networkError instanceof TypeError) {\n const isMissingKey = !appKey || !appKey.startsWith('pk_live_');\n const isLocalhost =\n typeof window !== 'undefined' &&\n (window.location.hostname === 'localhost' ||\n window.location.hostname === '127.0.0.1');\n\n if (isMissingKey && !isLocalhost) {\n throw new Error(\n 'Missing GitHat API key. Add NEXT_PUBLIC_GITHAT_PUBLISHABLE_KEY to .env.local'\n );\n }\n throw new Error('Unable to connect to GitHat API. Check your network connection.');\n }\n throw networkError;\n }\n\n if (response.status === 401) {\n // Queue all 401 retries behind a single refresh promise\n if (!_refreshPromise) {\n _refreshPromise = refreshTokens(apiUrl, appKey, useCookies).finally(() => {\n _refreshPromise = null;\n });\n }\n\n const refreshed = await _refreshPromise;\n\n if (refreshed) {\n // Get the new token (only needed in localStorage mode)\n const newToken =\n !useCookies && typeof window !== 'undefined'\n ? localStorage.getItem(TOKEN_KEYS.accessToken)\n : null;\n\n const retryResponse = await fetch(url, {\n ...fetchOptions,\n headers: {\n ...headers,\n ...(newToken && { Authorization: `Bearer ${newToken}` }),\n },\n credentials: useCookies ? 'include' : 'same-origin',\n });\n\n const retryData = await retryResponse.json();\n if (!retryResponse.ok) throw new Error(retryData.error || 'Request failed');\n return retryData as T;\n }\n\n clearAuth();\n throw new Error('Session expired');\n }\n\n const data = await response.json();\n if (!response.ok) throw new Error(data.error || 'Request failed');\n return data as T;\n }\n\n return { fetchApi, clearAuth };\n}\n","'use client';\n\nimport { useContext, useMemo, useCallback } from 'react';\nimport { GitHatContext } from './provider';\nimport { createClient } from './client';\nimport type { GitHatContextValue, GitHatOrg } from './types';\n\nexport interface OrgMetadata {\n [key: string]: unknown;\n}\n\nexport function useAuth(): GitHatContextValue {\n const ctx = useContext(GitHatContext);\n if (!ctx) throw new Error('useAuth must be used within a <GitHatProvider>');\n return ctx;\n}\n\nexport function useGitHat() {\n const ctx = useAuth();\n const client = useMemo(\n () => createClient(ctx.config.apiUrl!, ctx.config.publishableKey),\n [ctx.config.apiUrl, ctx.config.publishableKey]\n );\n\n /**\n * Get the current organization's metadata.\n * Requires the user to be signed in with an active organization.\n *\n * @example\n * ```tsx\n * const { getOrgMetadata } = useGitHat();\n * const meta = await getOrgMetadata();\n * console.log(meta.stripeAccountId);\n * ```\n */\n const getOrgMetadata = useCallback(async (): Promise<OrgMetadata> => {\n if (!ctx.org?.id) {\n throw new Error('No active organization');\n }\n const response = await client.fetchApi<{ metadata: OrgMetadata }>(\n `/orgs/${ctx.org.id}/metadata`\n );\n return response.metadata || {};\n }, [client, ctx.org?.id]);\n\n /**\n * Update the current organization's metadata.\n * Merges the provided object with existing metadata.\n * Set a key to null to delete it.\n * Requires admin or owner role.\n *\n * @example\n * ```tsx\n * const { updateOrgMetadata } = useGitHat();\n * // Set values\n * await updateOrgMetadata({ stripeAccountId: 'acct_xxx', features: ['pos'] });\n * // Delete a key\n * await updateOrgMetadata({ oldKey: null });\n * ```\n */\n const updateOrgMetadata = useCallback(\n async (updates: OrgMetadata): Promise<OrgMetadata> => {\n if (!ctx.org?.id) {\n throw new Error('No active organization');\n }\n const response = await client.fetchApi<{ metadata: OrgMetadata }>(\n `/orgs/${ctx.org.id}/metadata`,\n {\n method: 'PATCH',\n body: JSON.stringify(updates),\n }\n );\n return response.metadata || {};\n },\n [client, ctx.org?.id]\n );\n\n return {\n fetch: client.fetchApi,\n getUserOrgs: () => client.fetchApi<{ orgs: GitHatOrg[] }>('/user/orgs'),\n verifyMCP: (domain: string) => client.fetchApi<{ verified: boolean }>(`/verify/mcp/${domain}`),\n verifyAgent: (wallet: string) => client.fetchApi<{ verified: boolean }>(`/verify/agent/${wallet}`),\n getOrgMetadata,\n updateOrgMetadata,\n };\n}\n","'use client';\n\nimport { useMemo } from 'react';\nimport { useAuth } from './hooks';\nimport { createClient } from './client';\n\nexport interface DataItem {\n id: string;\n [key: string]: unknown;\n _createdAt?: string;\n _updatedAt?: string;\n}\n\nexport interface QueryOptions {\n limit?: number;\n cursor?: string;\n filter?: Record<string, unknown>;\n}\n\nexport interface QueryResult<T = DataItem> {\n items: T[];\n collection: string;\n nextCursor: string | null;\n count: number;\n}\n\nexport interface PutResult<T = DataItem> {\n item: T;\n collection: string;\n created: boolean;\n}\n\nexport interface DeleteResult {\n deleted: boolean;\n id: string;\n collection: string;\n}\n\nexport interface BatchOperation {\n type: 'put' | 'delete';\n id: string;\n data?: Record<string, unknown>;\n}\n\nexport interface BatchResult {\n processed: number;\n put: number;\n deleted: number;\n collection: string;\n}\n\n/**\n * Hook for interacting with GitHat's Customer Data API.\n * Provides CRUD operations for storing app data in GitHat's managed DynamoDB.\n *\n * @example\n * ```tsx\n * const { put, get, query, remove, batch } = useData();\n *\n * // Store data\n * await put('orders', { id: 'order_123', amount: 99.99, status: 'pending' });\n *\n * // Get single item\n * const order = await get('orders', 'order_123');\n *\n * // Query collection\n * const { items } = await query('orders', { filter: { status: 'pending' } });\n *\n * // Delete item\n * await remove('orders', 'order_123');\n * ```\n */\nexport function useData() {\n const ctx = useAuth();\n const client = useMemo(\n () => createClient(ctx.config.apiUrl!, ctx.config.publishableKey),\n [ctx.config.apiUrl, ctx.config.publishableKey]\n );\n\n return useMemo(() => ({\n /**\n * Store an item in a collection. If the item exists, it will be updated.\n * @param collection - Collection name (e.g., 'orders', 'users')\n * @param data - Data object with required `id` field\n */\n put: async <T extends DataItem>(collection: string, data: T): Promise<PutResult<T>> => {\n if (!data.id) {\n throw new Error('Data must include an \"id\" field');\n }\n return client.fetchApi<PutResult<T>>(`/data/${collection}/${data.id}`, {\n method: 'PUT',\n body: JSON.stringify(data),\n });\n },\n\n /**\n * Get a single item from a collection.\n * @param collection - Collection name\n * @param id - Item ID\n */\n get: async <T extends DataItem>(collection: string, id: string): Promise<T | null> => {\n try {\n const result = await client.fetchApi<{ item: T }>(`/data/${collection}/${id}`);\n return result.item;\n } catch (err: unknown) {\n if (err instanceof Error && err.message === 'Item not found') {\n return null;\n }\n throw err;\n }\n },\n\n /**\n * Query items from a collection with optional filters and pagination.\n * @param collection - Collection name\n * @param options - Query options (limit, cursor, filter)\n */\n query: async <T extends DataItem>(\n collection: string,\n options: QueryOptions = {}\n ): Promise<QueryResult<T>> => {\n const params = new URLSearchParams();\n if (options.limit) params.set('limit', options.limit.toString());\n if (options.cursor) params.set('cursor', options.cursor);\n if (options.filter) params.set('filter', JSON.stringify(options.filter));\n\n const queryString = params.toString();\n const url = `/data/${collection}${queryString ? `?${queryString}` : ''}`;\n\n return client.fetchApi<QueryResult<T>>(url);\n },\n\n /**\n * Delete an item from a collection.\n * @param collection - Collection name\n * @param id - Item ID\n */\n remove: async (collection: string, id: string): Promise<DeleteResult> => {\n return client.fetchApi<DeleteResult>(`/data/${collection}/${id}`, {\n method: 'DELETE',\n });\n },\n\n /**\n * Batch operations (put/delete) on a collection.\n * Maximum 100 operations per request.\n * @param collection - Collection name\n * @param operations - Array of operations\n */\n batch: async (collection: string, operations: BatchOperation[]): Promise<BatchResult> => {\n return client.fetchApi<BatchResult>(`/data/${collection}/batch`, {\n method: 'POST',\n body: JSON.stringify({ operations }),\n });\n },\n }), [client]);\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useAuth } from '../hooks';\n\ninterface SignInFormProps {\n onSuccess?: () => void;\n signUpUrl?: string;\n forgotPasswordUrl?: string;\n}\n\nexport function SignInForm({ onSuccess, signUpUrl, forgotPasswordUrl }: SignInFormProps) {\n const { signIn, config } = useAuth();\n const [email, setEmail] = useState('');\n const [password, setPassword] = useState('');\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n const emailValid = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(email);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (!emailValid) {\n setError('Please enter a valid email address');\n return;\n }\n setError('');\n setLoading(true);\n try {\n await signIn(email, password);\n if (onSuccess) {\n onSuccess();\n } else if (typeof window !== 'undefined') {\n const params = new URLSearchParams(window.location.search);\n window.location.href = params.get('redirect_url') || config.afterSignInUrl!;\n }\n } catch (err: any) {\n setError(err.message || 'Sign in failed');\n } finally {\n setLoading(false);\n }\n };\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Sign in</h2>\n <p className=\"githat-form-subtitle\">Welcome back to GitHat</p>\n </div>\n {error && <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">{error}</div>}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Sign in form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signin-email\">Email</label>\n <input\n id=\"githat-signin-email\"\n className=\"githat-input\"\n type=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n placeholder=\"you@example.com\"\n autoComplete=\"email\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signin-password\">Password</label>\n <input\n id=\"githat-signin-password\"\n className=\"githat-input\"\n type=\"password\"\n value={password}\n onChange={(e) => setPassword(e.target.value)}\n placeholder=\"Enter your password\"\n autoComplete=\"current-password\"\n required\n />\n </div>\n {forgotPasswordUrl && (\n <a href={forgotPasswordUrl} className=\"githat-link githat-forgot-link\">Forgot password?</a>\n )}\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={loading || !email || !password || (email.length > 0 && !emailValid)}\n >\n {loading ? 'Signing in...' : 'Sign in'}\n </button>\n </form>\n {signUpUrl && (\n <p className=\"githat-form-footer\">\n Don't have an account? <a href={signUpUrl} className=\"githat-link\">Sign up</a>\n </p>\n )}\n <p className=\"githat-powered-by\">Secured by <strong>GitHat</strong></p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useAuth } from '../hooks';\n\ninterface SignUpFormProps {\n onSuccess?: (result: { requiresVerification: boolean; email: string }) => void;\n signInUrl?: string;\n}\n\nexport function SignUpForm({ onSuccess, signInUrl }: SignUpFormProps) {\n const { signUp, config } = useAuth();\n const [name, setName] = useState('');\n const [email, setEmail] = useState('');\n const [password, setPassword] = useState('');\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n const emailValid = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(email);\n const passwordValid = password.length >= 8\n && /[A-Z]/.test(password)\n && /[a-z]/.test(password)\n && /\\d/.test(password);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (!emailValid) {\n setError('Please enter a valid email address');\n return;\n }\n if (!passwordValid) {\n setError('Password must be 8+ characters with uppercase, lowercase, number, and special character');\n return;\n }\n setError('');\n setLoading(true);\n try {\n const result = await signUp({ email, password, name });\n if (onSuccess) {\n onSuccess(result);\n } else if (typeof window !== 'undefined') {\n window.location.href = config.afterSignInUrl!;\n }\n } catch (err: any) {\n setError(err.message || 'Sign up failed');\n } finally {\n setLoading(false);\n }\n };\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Create an account</h2>\n <p className=\"githat-form-subtitle\">Get started with GitHat</p>\n </div>\n {error && <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">{error}</div>}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Sign up form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-name\">Full name</label>\n <input\n id=\"githat-signup-name\"\n className=\"githat-input\"\n type=\"text\"\n value={name}\n onChange={(e) => setName(e.target.value)}\n placeholder=\"Your name\"\n autoComplete=\"name\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-email\">Email</label>\n <input\n id=\"githat-signup-email\"\n className=\"githat-input\"\n type=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n placeholder=\"you@example.com\"\n autoComplete=\"email\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-password\">Password</label>\n <input\n id=\"githat-signup-password\"\n className=\"githat-input\"\n type=\"password\"\n value={password}\n onChange={(e) => setPassword(e.target.value)}\n placeholder=\"8+ characters\"\n autoComplete=\"new-password\"\n required\n />\n {password && !passwordValid && (\n <p className=\"githat-field-error\">\n Must be 8+ characters with uppercase, lowercase, and number\n </p>\n )}\n </div>\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={loading || !email || !password || !name || !emailValid}\n >\n {loading ? 'Creating account...' : 'Sign up'}\n </button>\n </form>\n {signInUrl && (\n <p className=\"githat-form-footer\">\n Already have an account? <a href={signInUrl} className=\"githat-link\">Sign in</a>\n </p>\n )}\n <p className=\"githat-powered-by\">Secured by <strong>GitHat</strong></p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useContext } from 'react';\nimport { GitHatContext } from '../provider';\n\ninterface SignInButtonProps {\n className?: string;\n children?: React.ReactNode;\n href?: string;\n}\n\nexport function SignInButton({ className, children, href }: SignInButtonProps) {\n const ctx = useContext(GitHatContext);\n const url = href || ctx?.config.signInUrl || '/sign-in';\n return (\n <a href={url} className={className || 'githat-button githat-button-primary'} aria-label=\"Sign in\">\n {children || 'Sign in'}\n </a>\n );\n}\n","'use client';\n\nimport React, { useContext } from 'react';\nimport { GitHatContext } from '../provider';\n\ninterface SignUpButtonProps {\n className?: string;\n children?: React.ReactNode;\n href?: string;\n}\n\nexport function SignUpButton({ className, children, href }: SignUpButtonProps) {\n const ctx = useContext(GitHatContext);\n const url = href || ctx?.config.signUpUrl || '/sign-up';\n return (\n <a href={url} className={className || 'githat-button githat-button-outline'} aria-label=\"Sign up\">\n {children || 'Sign up'}\n </a>\n );\n}\n","'use client';\n\nimport React, { useState, useRef, useEffect } from 'react';\nimport { useAuth } from '../hooks';\n\nexport function UserButton() {\n const { user, org, isSignedIn, signOut } = useAuth();\n const [open, setOpen] = useState(false);\n const ref = useRef<HTMLDivElement>(null);\n\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (ref.current && !ref.current.contains(e.target as Node)) setOpen(false);\n };\n document.addEventListener('mousedown', handleClickOutside);\n return () => document.removeEventListener('mousedown', handleClickOutside);\n }, []);\n\n if (!isSignedIn || !user) return null;\n\n const initials = user.name\n ? user.name.split(' ').map(n => n[0]).join('').toUpperCase().slice(0, 2)\n : user.email[0].toUpperCase();\n\n return (\n <div className=\"githat-user-button\" ref={ref}>\n <button className=\"githat-avatar-trigger\" onClick={() => setOpen(!open)} aria-label=\"User menu\" aria-expanded={open} aria-haspopup=\"true\">\n {user.avatarUrl ? (\n <img src={user.avatarUrl} alt={user.name || 'User avatar'} className=\"githat-avatar-img\" />\n ) : (\n <span className=\"githat-avatar-initials\">{initials}</span>\n )}\n </button>\n {open && (\n <div className=\"githat-dropdown\" role=\"menu\">\n <div className=\"githat-dropdown-header\">\n <p className=\"githat-dropdown-name\">{user.name}</p>\n <p className=\"githat-dropdown-email\">{user.email}</p>\n {org && <p className=\"githat-dropdown-org\">{org.name}</p>}\n </div>\n <div className=\"githat-dropdown-divider\" />\n <button className=\"githat-dropdown-item\" role=\"menuitem\" onClick={() => { signOut(); setOpen(false); }}>\n Sign out\n </button>\n </div>\n )}\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, useEffect, useRef } from 'react';\nimport { useAuth, useGitHat } from '../hooks';\nimport type { GitHatOrg } from '../types';\n\nexport function OrgSwitcher() {\n const { org, isSignedIn, switchOrg } = useAuth();\n const githat = useGitHat();\n const [orgs, setOrgs] = useState<GitHatOrg[]>([]);\n const [orgsLoading, setOrgsLoading] = useState(false);\n const [open, setOpen] = useState(false);\n const ref = useRef<HTMLDivElement>(null);\n\n useEffect(() => {\n if (isSignedIn) {\n setOrgsLoading(true);\n githat.getUserOrgs()\n .then(data => setOrgs(data.orgs || []))\n .catch(() => {})\n .finally(() => setOrgsLoading(false));\n }\n }, [isSignedIn]);\n\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (ref.current && !ref.current.contains(e.target as Node)) setOpen(false);\n };\n document.addEventListener('mousedown', handleClickOutside);\n return () => document.removeEventListener('mousedown', handleClickOutside);\n }, []);\n\n if (!isSignedIn || !org || (orgs.length < 2 && !orgsLoading)) return null;\n\n return (\n <div className=\"githat-org-switcher\" ref={ref}>\n <button className=\"githat-org-trigger\" onClick={() => setOpen(!open)} aria-label=\"Switch organization\" aria-expanded={open} aria-haspopup=\"true\">\n <span className=\"githat-org-name\">{org.name}</span>\n <span className=\"githat-chevron\">{open ? '\\u25B2' : '\\u25BC'}</span>\n </button>\n {open && (\n <div className=\"githat-dropdown\" role=\"menu\">\n {orgsLoading ? (\n <div className=\"githat-dropdown-item\" aria-busy=\"true\">Loading...</div>\n ) : orgs.map(o => (\n <button\n key={o.id}\n className={`githat-dropdown-item ${o.id === org.id ? 'githat-dropdown-item-active' : ''}`}\n role=\"menuitem\"\n aria-current={o.id === org.id ? 'true' : undefined}\n onClick={() => { switchOrg(o.id); setOpen(false); }}\n >\n {o.name}\n {o.id === org.id && <span className=\"githat-check\">{'\\u2713'}</span>}\n </button>\n ))}\n </div>\n )}\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, useEffect, useRef } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface VerifiedBadgeProps {\n type: 'mcp' | 'agent';\n identifier: string;\n label?: string;\n}\n\nconst CACHE_TTL = 5 * 60 * 1000; // 5 minutes\nconst cache = new Map<string, { verified: boolean; ts: number }>();\n\nexport function VerifiedBadge({ type, identifier, label }: VerifiedBadgeProps) {\n const githat = useGitHat();\n const [verified, setVerified] = useState<boolean | null>(null);\n const mounted = useRef(true);\n\n useEffect(() => {\n mounted.current = true;\n const key = `${type}:${identifier}`;\n const cached = cache.get(key);\n if (cached && Date.now() - cached.ts < CACHE_TTL) {\n setVerified(cached.verified);\n return;\n }\n\n const verify = type === 'mcp' ? githat.verifyMCP : githat.verifyAgent;\n verify(identifier)\n .then(data => {\n if (mounted.current) {\n setVerified(data.verified);\n cache.set(key, { verified: data.verified, ts: Date.now() });\n }\n })\n .catch(() => { if (mounted.current) setVerified(false); });\n\n return () => { mounted.current = false; };\n }, [type, identifier]);\n\n if (verified === null) return null;\n\n return (\n <span className={`githat-badge ${verified ? 'githat-badge-verified' : 'githat-badge-unverified'}`}>\n {verified ? '\\u2713' : '\\u2717'} {label || (verified ? 'Verified' : 'Unverified')}\n </span>\n );\n}\n","'use client';\n\nimport React from 'react';\nimport { useAuth } from '../hooks';\n\ninterface ProtectedRouteProps {\n children: React.ReactNode;\n fallback?: React.ReactNode;\n}\n\nexport function ProtectedRoute({ children, fallback }: ProtectedRouteProps) {\n const { isSignedIn, isLoading, config } = useAuth();\n\n if (isLoading) {\n return <>{fallback || <div className=\"githat-loading\">Loading...</div>}</>;\n }\n\n if (!isSignedIn) {\n if (typeof window !== 'undefined') {\n window.location.href = config.signInUrl!;\n }\n return null;\n }\n\n return <>{children}</>;\n}\n"],"mappings":"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","names":["useMemo","useCallback","useMemo","useCallback","useMemo","useMemo","useState","jsx","useState","useState","jsx","jsxs","useState","useContext","jsx","useContext","useContext","jsx","useContext","useState","useRef","useEffect","jsx","jsxs","useState","useRef","useEffect","useState","useEffect","useRef","jsx","jsxs","useState","useRef","useEffect","useState","useEffect","useRef","jsxs","useState","useRef","useEffect","jsx"]}
1
+ {"version":3,"sources":["../src/provider.tsx","../src/config.ts","../src/client.ts","../src/hooks.ts","../src/data.ts","../src/components/SignInForm.tsx","../src/components/SignUpForm.tsx","../src/components/SignInButton.tsx","../src/components/SignUpButton.tsx","../src/components/UserButton.tsx","../src/components/OrgSwitcher.tsx","../src/components/VerifiedBadge.tsx","../src/components/ProtectedRoute.tsx","../src/components/ForgotPasswordForm.tsx","../src/components/ResetPasswordForm.tsx","../src/components/VerifyEmailStatus.tsx","../src/components/ChangePasswordForm.tsx"],"sourcesContent":["'use client';\n\nimport React, { createContext, useState, useEffect, useCallback, useMemo, useRef } from 'react';\nimport type { GitHatConfig, GitHatContextValue, GitHatUser, GitHatOrg, SignUpData, SignUpResult } from './types';\nimport { TOKEN_KEYS, resolveConfig } from './config';\nimport { createClient } from './client';\n\nexport const GitHatContext = createContext<GitHatContextValue | null>(null);\n\ninterface GitHatProviderProps {\n config: GitHatConfig;\n children: React.ReactNode;\n}\n\nexport function GitHatProvider({ config: rawConfig, children }: GitHatProviderProps) {\n const config = useMemo(() => resolveConfig(rawConfig), [rawConfig]);\n const useCookies = config.tokenStorage === 'cookie';\n\n // Create client with cookie mode awareness\n const clientRef = useRef(createClient(config.apiUrl, config.publishableKey, { useCookies }));\n\n const [user, setUser] = useState<GitHatUser | null>(null);\n const [org, setOrg] = useState<GitHatOrg | null>(null);\n const [isSignedIn, setIsSignedIn] = useState(false);\n const [isLoading, setIsLoading] = useState(true);\n const [authError, setAuthError] = useState<string | null>(null);\n\n // Validate stored token/session on mount\n useEffect(() => {\n const validateSession = async () => {\n try {\n // In cookie mode, we always try to validate (cookies sent automatically)\n // In localStorage mode, only validate if we have a token\n if (!useCookies) {\n const token = localStorage.getItem(TOKEN_KEYS.accessToken);\n const storedUser = localStorage.getItem(TOKEN_KEYS.user);\n if (!token || !storedUser) {\n setIsLoading(false);\n return;\n }\n }\n\n const data = await clientRef.current.fetchApi<{ user: GitHatUser; currentOrg: GitHatOrg }>('/auth/me');\n\n if (data.user) {\n setUser(data.user);\n setOrg(data.currentOrg || null);\n setIsSignedIn(true);\n setAuthError(null);\n\n // Store user info in localStorage for client-side access (even in cookie mode)\n if (!useCookies) {\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.currentOrg) {\n localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.currentOrg));\n }\n }\n }\n } catch (err: unknown) {\n const error = err as Error;\n if (error.message === 'Session expired') {\n clientRef.current.clearAuth();\n } else if (!useCookies) {\n // Network error in localStorage mode — try to use stored user\n const storedUser = localStorage.getItem(TOKEN_KEYS.user);\n if (storedUser) {\n try {\n setUser(JSON.parse(storedUser));\n setIsSignedIn(true);\n } catch {}\n }\n setAuthError(error.message || 'Failed to verify session');\n }\n // In cookie mode with error, just stay logged out\n }\n setIsLoading(false);\n };\n\n validateSession();\n }, [useCookies]);\n\n // Listen for auth-changed events\n useEffect(() => {\n const handleAuthChanged = (e: Event) => {\n const detail = (e as CustomEvent).detail;\n if (detail?.signedIn === false) {\n setIsSignedIn(false);\n setUser(null);\n setOrg(null);\n } else if (detail?.signedIn === true && detail?.user) {\n setUser(detail.user);\n setIsSignedIn(true);\n if (detail.org) setOrg(detail.org);\n }\n };\n window.addEventListener('githat:auth-changed', handleAuthChanged);\n return () => window.removeEventListener('githat:auth-changed', handleAuthChanged);\n }, []);\n\n const signIn = useCallback(async (email: string, password: string) => {\n // Build login URL with setCookie param if in cookie mode\n const loginUrl = useCookies ? '/auth/login?setCookie=true' : '/auth/login';\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n user: GitHatUser;\n org: GitHatOrg;\n }>(loginUrl, {\n method: 'POST',\n body: JSON.stringify({ email, password }),\n });\n\n // In localStorage mode, store tokens\n if (!useCookies && data.accessToken && data.refreshToken) {\n localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setUser(data.user);\n setOrg(data.org || null);\n setIsSignedIn(true);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user: data.user, org: data.org, signedIn: true },\n }));\n }, [useCookies]);\n\n const signUp = useCallback(async (signUpData: SignUpData): Promise<SignUpResult> => {\n // Build register URL with setCookie param if in cookie mode\n const registerUrl = useCookies ? '/auth/register?setCookie=true' : '/auth/register';\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n user: GitHatUser;\n org: GitHatOrg;\n }>(registerUrl, {\n method: 'POST',\n body: JSON.stringify(signUpData),\n });\n\n // In localStorage mode, store tokens\n if (!useCookies && data.accessToken && data.refreshToken) {\n localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.user, JSON.stringify(data.user));\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setUser(data.user);\n setOrg(data.org || null);\n setIsSignedIn(true);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user: data.user, org: data.org, signedIn: true },\n }));\n\n return { requiresVerification: !data.user.emailVerified, email: signUpData.email };\n }, [useCookies]);\n\n const signOut = useCallback(async () => {\n try {\n // In cookie mode, logout endpoint clears cookies\n const logoutUrl = useCookies ? '/auth/logout?setCookie=true' : '/auth/logout';\n await clientRef.current.fetchApi(logoutUrl, { method: 'POST' });\n } catch {}\n clientRef.current.clearAuth();\n setIsSignedIn(false);\n setUser(null);\n setOrg(null);\n if (typeof window !== 'undefined' && config.afterSignOutUrl) {\n window.location.href = config.afterSignOutUrl;\n }\n }, [config.afterSignOutUrl, useCookies]);\n\n const switchOrg = useCallback(async (orgId: string) => {\n try {\n const switchUrl = useCookies\n ? `/user/orgs/${orgId}/switch?setCookie=true`\n : `/user/orgs/${orgId}/switch`;\n\n const data = await clientRef.current.fetchApi<{\n accessToken?: string;\n refreshToken?: string;\n org: GitHatOrg;\n }>(switchUrl, { method: 'POST' });\n\n // In localStorage mode, store new tokens\n if (!useCookies) {\n if (data.accessToken) localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n if (data.refreshToken) localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n }\n\n setOrg(data.org);\n\n window.dispatchEvent(new CustomEvent('githat:auth-changed', {\n detail: { user, org: data.org, signedIn: true },\n }));\n } catch (e) {\n console.error('Org switch failed:', e);\n }\n }, [user, useCookies]);\n\n const value = useMemo<GitHatContextValue>(() => ({\n user, org, isSignedIn, isLoading, authError, config,\n signIn, signUp, signOut, switchOrg,\n }), [user, org, isSignedIn, isLoading, authError, config, signIn, signUp, signOut, switchOrg]);\n\n return (\n <GitHatContext.Provider value={value}>\n {children}\n </GitHatContext.Provider>\n );\n}\n","import type { GitHatConfig } from './types';\n\nexport const DEFAULT_API_URL = 'https://api.githat.io';\n\nexport const TOKEN_KEYS = {\n accessToken: 'githat_access_token',\n refreshToken: 'githat_refresh_token',\n user: 'githat_user',\n org: 'githat_org',\n} as const;\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\nexport function resolveConfig(config: GitHatConfig): Required<GitHatConfig> {\n return {\n publishableKey: config.publishableKey,\n apiUrl: config.apiUrl || DEFAULT_API_URL,\n signInUrl: config.signInUrl || '/sign-in',\n signUpUrl: config.signUpUrl || '/sign-up',\n afterSignInUrl: config.afterSignInUrl || '/dashboard',\n afterSignOutUrl: config.afterSignOutUrl || '/',\n tokenStorage: config.tokenStorage || 'localStorage',\n };\n}\n","import { TOKEN_KEYS } from './config';\n\ninterface ClientOptions {\n /**\n * When true, use credentials: 'include' for all requests\n * and don't send Authorization header (cookies handle auth)\n */\n useCookies?: boolean;\n}\n\nlet _refreshPromise: Promise<boolean> | null = null;\n\nasync function refreshTokens(\n apiUrl: string,\n appKey: string,\n useCookies: boolean\n): Promise<boolean> {\n // In cookie mode, refresh token comes from cookie\n // In localStorage mode, we need to send it in the body\n const refreshToken =\n typeof window !== 'undefined' && !useCookies\n ? localStorage.getItem(TOKEN_KEYS.refreshToken)\n : null;\n\n // In localStorage mode, we need a refresh token\n if (!useCookies && !refreshToken) return false;\n\n let orgId: string | null = null;\n try {\n const orgStr = localStorage.getItem(TOKEN_KEYS.org);\n if (orgStr) orgId = JSON.parse(orgStr).id;\n } catch {}\n\n try {\n const refreshUrl = useCookies\n ? `${apiUrl}/auth/refresh?setCookie=true`\n : `${apiUrl}/auth/refresh`;\n\n const res = await fetch(refreshUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-GitHat-App-Key': appKey,\n },\n credentials: useCookies ? 'include' : 'same-origin',\n body: JSON.stringify(useCookies ? { orgId } : { refreshToken, orgId }),\n });\n\n if (!res.ok) return false;\n\n const data = await res.json();\n\n // In localStorage mode, store the new tokens\n if (!useCookies) {\n if (data.accessToken) localStorage.setItem(TOKEN_KEYS.accessToken, data.accessToken);\n if (data.refreshToken) localStorage.setItem(TOKEN_KEYS.refreshToken, data.refreshToken);\n }\n\n if (data.org) localStorage.setItem(TOKEN_KEYS.org, JSON.stringify(data.org));\n return true;\n } catch {\n return false;\n }\n}\n\nfunction clearAuth() {\n if (typeof window === 'undefined') return;\n Object.values(TOKEN_KEYS).forEach((key) => localStorage.removeItem(key));\n window.dispatchEvent(\n new CustomEvent('githat:auth-changed', {\n detail: { user: null, org: null, signedIn: false },\n })\n );\n}\n\nexport function createClient(\n apiUrl: string,\n appKey: string,\n options: ClientOptions = {}\n) {\n const { useCookies = false } = options;\n\n async function fetchApi<T = unknown>(\n endpoint: string,\n fetchOptions: RequestInit = {}\n ): Promise<T> {\n const url = `${apiUrl}${endpoint}`;\n\n // In localStorage mode, get token from storage\n // In cookie mode, cookies are sent automatically with credentials: 'include'\n const token =\n typeof window !== 'undefined' && !useCookies\n ? localStorage.getItem(TOKEN_KEYS.accessToken)\n : null;\n\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n 'X-GitHat-App-Key': appKey,\n ...(token && { Authorization: `Bearer ${token}` }),\n ...(fetchOptions.headers as Record<string, string>),\n };\n\n let response: Response;\n try {\n response = await fetch(url, {\n ...fetchOptions,\n headers,\n credentials: useCookies ? 'include' : 'same-origin',\n });\n } catch (networkError: unknown) {\n // Network errors (CORS blocked, offline, DNS failure) throw TypeError\n if (networkError instanceof TypeError) {\n const isMissingKey = !appKey || !appKey.startsWith('pk_live_');\n const isLocalhost =\n typeof window !== 'undefined' &&\n (window.location.hostname === 'localhost' ||\n window.location.hostname === '127.0.0.1');\n\n if (isMissingKey && !isLocalhost) {\n throw new Error(\n 'Missing GitHat API key. Add NEXT_PUBLIC_GITHAT_PUBLISHABLE_KEY to .env.local'\n );\n }\n throw new Error('Unable to connect to GitHat API. Check your network connection.');\n }\n throw networkError;\n }\n\n if (response.status === 401) {\n // Queue all 401 retries behind a single refresh promise\n if (!_refreshPromise) {\n _refreshPromise = refreshTokens(apiUrl, appKey, useCookies).finally(() => {\n _refreshPromise = null;\n });\n }\n\n const refreshed = await _refreshPromise;\n\n if (refreshed) {\n // Get the new token (only needed in localStorage mode)\n const newToken =\n !useCookies && typeof window !== 'undefined'\n ? localStorage.getItem(TOKEN_KEYS.accessToken)\n : null;\n\n const retryResponse = await fetch(url, {\n ...fetchOptions,\n headers: {\n ...headers,\n ...(newToken && { Authorization: `Bearer ${newToken}` }),\n },\n credentials: useCookies ? 'include' : 'same-origin',\n });\n\n const retryData = await retryResponse.json();\n if (!retryResponse.ok) throw new Error(retryData.error || 'Request failed');\n return retryData as T;\n }\n\n clearAuth();\n throw new Error('Session expired');\n }\n\n const data = await response.json();\n if (!response.ok) throw new Error(data.error || 'Request failed');\n return data as T;\n }\n\n return { fetchApi, clearAuth };\n}\n","'use client';\n\nimport { useContext, useMemo, useCallback } from 'react';\nimport { GitHatContext } from './provider';\nimport { createClient } from './client';\nimport type { GitHatContextValue, GitHatOrg } from './types';\n\nexport interface OrgMetadata {\n [key: string]: unknown;\n}\n\nexport function useAuth(): GitHatContextValue {\n const ctx = useContext(GitHatContext);\n if (!ctx) throw new Error('useAuth must be used within a <GitHatProvider>');\n return ctx;\n}\n\nexport function useGitHat() {\n const ctx = useAuth();\n const client = useMemo(\n () => createClient(ctx.config.apiUrl!, ctx.config.publishableKey),\n [ctx.config.apiUrl, ctx.config.publishableKey]\n );\n\n /**\n * Get the current organization's metadata.\n * Requires the user to be signed in with an active organization.\n *\n * @example\n * ```tsx\n * const { getOrgMetadata } = useGitHat();\n * const meta = await getOrgMetadata();\n * console.log(meta.stripeAccountId);\n * ```\n */\n const getOrgMetadata = useCallback(async (): Promise<OrgMetadata> => {\n if (!ctx.org?.id) {\n throw new Error('No active organization');\n }\n const response = await client.fetchApi<{ metadata: OrgMetadata }>(\n `/orgs/${ctx.org.id}/metadata`\n );\n return response.metadata || {};\n }, [client, ctx.org?.id]);\n\n /**\n * Update the current organization's metadata.\n * Merges the provided object with existing metadata.\n * Set a key to null to delete it.\n * Requires admin or owner role.\n *\n * @example\n * ```tsx\n * const { updateOrgMetadata } = useGitHat();\n * // Set values\n * await updateOrgMetadata({ stripeAccountId: 'acct_xxx', features: ['pos'] });\n * // Delete a key\n * await updateOrgMetadata({ oldKey: null });\n * ```\n */\n const updateOrgMetadata = useCallback(\n async (updates: OrgMetadata): Promise<OrgMetadata> => {\n if (!ctx.org?.id) {\n throw new Error('No active organization');\n }\n const response = await client.fetchApi<{ metadata: OrgMetadata }>(\n `/orgs/${ctx.org.id}/metadata`,\n {\n method: 'PATCH',\n body: JSON.stringify(updates),\n }\n );\n return response.metadata || {};\n },\n [client, ctx.org?.id]\n );\n\n // ============================================================================\n // Password Management\n // ============================================================================\n\n /**\n * Request a password reset email.\n *\n * @example\n * ```tsx\n * const { forgotPassword } = useGitHat();\n * await forgotPassword('user@example.com');\n * // Check email for reset link\n * ```\n */\n const forgotPassword = useCallback(\n async (email: string): Promise<{ success: boolean }> => {\n const response = await fetch(`${ctx.config.apiUrl}/auth/forgot-password`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ email }),\n });\n if (!response.ok) {\n const error = await response.json().catch(() => ({}));\n throw new Error(error.message || 'Failed to send reset email');\n }\n return { success: true };\n },\n [ctx.config.apiUrl]\n );\n\n /**\n * Reset password using a token from email.\n *\n * @example\n * ```tsx\n * const { resetPassword } = useGitHat();\n * const token = searchParams.get('token');\n * await resetPassword(token, 'NewSecurePassword123!');\n * ```\n */\n const resetPassword = useCallback(\n async (token: string, newPassword: string): Promise<{ success: boolean }> => {\n const response = await fetch(`${ctx.config.apiUrl}/auth/reset-password`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ token, password: newPassword }),\n });\n if (!response.ok) {\n const error = await response.json().catch(() => ({}));\n throw new Error(error.message || 'Failed to reset password');\n }\n return { success: true };\n },\n [ctx.config.apiUrl]\n );\n\n /**\n * Change password for authenticated user.\n * Requires current password for verification.\n *\n * @example\n * ```tsx\n * const { changePassword } = useGitHat();\n * await changePassword('currentPassword', 'newSecurePassword123!');\n * ```\n */\n const changePassword = useCallback(\n async (currentPassword: string, newPassword: string): Promise<{ success: boolean }> => {\n await client.fetchApi('/auth/change-password', {\n method: 'POST',\n body: JSON.stringify({ currentPassword, newPassword }),\n });\n return { success: true };\n },\n [client]\n );\n\n // ============================================================================\n // Email Verification\n // ============================================================================\n\n /**\n * Verify email using a token from the verification email.\n *\n * @example\n * ```tsx\n * const { verifyEmail } = useGitHat();\n * const token = searchParams.get('token');\n * await verifyEmail(token);\n * ```\n */\n const verifyEmail = useCallback(\n async (token: string): Promise<{ success: boolean }> => {\n const response = await fetch(`${ctx.config.apiUrl}/auth/verify-email`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ token }),\n });\n if (!response.ok) {\n const error = await response.json().catch(() => ({}));\n throw new Error(error.message || 'Failed to verify email');\n }\n return { success: true };\n },\n [ctx.config.apiUrl]\n );\n\n /**\n * Resend verification email to the specified address.\n *\n * @example\n * ```tsx\n * const { resendVerificationEmail } = useGitHat();\n * await resendVerificationEmail('user@example.com');\n * ```\n */\n const resendVerificationEmail = useCallback(\n async (email: string): Promise<{ success: boolean }> => {\n const response = await fetch(`${ctx.config.apiUrl}/auth/resend-verification`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ email }),\n });\n if (!response.ok) {\n const error = await response.json().catch(() => ({}));\n throw new Error(error.message || 'Failed to resend verification email');\n }\n return { success: true };\n },\n [ctx.config.apiUrl]\n );\n\n return {\n fetch: client.fetchApi,\n getUserOrgs: () => client.fetchApi<{ orgs: GitHatOrg[] }>('/user/orgs'),\n verifyMCP: (domain: string) => client.fetchApi<{ verified: boolean }>(`/verify/mcp/${domain}`),\n verifyAgent: (wallet: string) => client.fetchApi<{ verified: boolean }>(`/verify/agent/${wallet}`),\n getOrgMetadata,\n updateOrgMetadata,\n // Password management\n forgotPassword,\n resetPassword,\n changePassword,\n // Email verification\n verifyEmail,\n resendVerificationEmail,\n };\n}\n","'use client';\n\nimport { useMemo } from 'react';\nimport { useAuth } from './hooks';\nimport { createClient } from './client';\n\nexport interface DataItem {\n id: string;\n [key: string]: unknown;\n _createdAt?: string;\n _updatedAt?: string;\n}\n\nexport interface QueryOptions {\n limit?: number;\n cursor?: string;\n filter?: Record<string, unknown>;\n}\n\nexport interface QueryResult<T = DataItem> {\n items: T[];\n collection: string;\n nextCursor: string | null;\n count: number;\n}\n\nexport interface PutResult<T = DataItem> {\n item: T;\n collection: string;\n created: boolean;\n}\n\nexport interface DeleteResult {\n deleted: boolean;\n id: string;\n collection: string;\n}\n\nexport interface BatchOperation {\n type: 'put' | 'delete';\n id: string;\n data?: Record<string, unknown>;\n}\n\nexport interface BatchResult {\n processed: number;\n put: number;\n deleted: number;\n collection: string;\n}\n\n/**\n * Hook for interacting with GitHat's Customer Data API.\n * Provides CRUD operations for storing app data in GitHat's managed DynamoDB.\n *\n * @example\n * ```tsx\n * const { put, get, query, remove, batch } = useData();\n *\n * // Store data\n * await put('orders', { id: 'order_123', amount: 99.99, status: 'pending' });\n *\n * // Get single item\n * const order = await get('orders', 'order_123');\n *\n * // Query collection\n * const { items } = await query('orders', { filter: { status: 'pending' } });\n *\n * // Delete item\n * await remove('orders', 'order_123');\n * ```\n */\nexport function useData() {\n const ctx = useAuth();\n const client = useMemo(\n () => createClient(ctx.config.apiUrl!, ctx.config.publishableKey),\n [ctx.config.apiUrl, ctx.config.publishableKey]\n );\n\n return useMemo(() => ({\n /**\n * Store an item in a collection. If the item exists, it will be updated.\n * @param collection - Collection name (e.g., 'orders', 'users')\n * @param data - Data object with required `id` field\n */\n put: async <T extends DataItem>(collection: string, data: T): Promise<PutResult<T>> => {\n if (!data.id) {\n throw new Error('Data must include an \"id\" field');\n }\n return client.fetchApi<PutResult<T>>(`/data/${collection}/${data.id}`, {\n method: 'PUT',\n body: JSON.stringify(data),\n });\n },\n\n /**\n * Get a single item from a collection.\n * @param collection - Collection name\n * @param id - Item ID\n */\n get: async <T extends DataItem>(collection: string, id: string): Promise<T | null> => {\n try {\n const result = await client.fetchApi<{ item: T }>(`/data/${collection}/${id}`);\n return result.item;\n } catch (err: unknown) {\n if (err instanceof Error && err.message === 'Item not found') {\n return null;\n }\n throw err;\n }\n },\n\n /**\n * Query items from a collection with optional filters and pagination.\n * @param collection - Collection name\n * @param options - Query options (limit, cursor, filter)\n */\n query: async <T extends DataItem>(\n collection: string,\n options: QueryOptions = {}\n ): Promise<QueryResult<T>> => {\n const params = new URLSearchParams();\n if (options.limit) params.set('limit', options.limit.toString());\n if (options.cursor) params.set('cursor', options.cursor);\n if (options.filter) params.set('filter', JSON.stringify(options.filter));\n\n const queryString = params.toString();\n const url = `/data/${collection}${queryString ? `?${queryString}` : ''}`;\n\n return client.fetchApi<QueryResult<T>>(url);\n },\n\n /**\n * Delete an item from a collection.\n * @param collection - Collection name\n * @param id - Item ID\n */\n remove: async (collection: string, id: string): Promise<DeleteResult> => {\n return client.fetchApi<DeleteResult>(`/data/${collection}/${id}`, {\n method: 'DELETE',\n });\n },\n\n /**\n * Batch operations (put/delete) on a collection.\n * Maximum 100 operations per request.\n * @param collection - Collection name\n * @param operations - Array of operations\n */\n batch: async (collection: string, operations: BatchOperation[]): Promise<BatchResult> => {\n return client.fetchApi<BatchResult>(`/data/${collection}/batch`, {\n method: 'POST',\n body: JSON.stringify({ operations }),\n });\n },\n }), [client]);\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useAuth } from '../hooks';\n\ninterface SignInFormProps {\n onSuccess?: () => void;\n signUpUrl?: string;\n forgotPasswordUrl?: string;\n}\n\nexport function SignInForm({ onSuccess, signUpUrl, forgotPasswordUrl }: SignInFormProps) {\n const { signIn, config } = useAuth();\n const [email, setEmail] = useState('');\n const [password, setPassword] = useState('');\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n const emailValid = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(email);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (!emailValid) {\n setError('Please enter a valid email address');\n return;\n }\n setError('');\n setLoading(true);\n try {\n await signIn(email, password);\n if (onSuccess) {\n onSuccess();\n } else if (typeof window !== 'undefined') {\n const params = new URLSearchParams(window.location.search);\n window.location.href = params.get('redirect_url') || config.afterSignInUrl!;\n }\n } catch (err: any) {\n setError(err.message || 'Sign in failed');\n } finally {\n setLoading(false);\n }\n };\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Sign in</h2>\n <p className=\"githat-form-subtitle\">Welcome back to GitHat</p>\n </div>\n {error && <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">{error}</div>}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Sign in form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signin-email\">Email</label>\n <input\n id=\"githat-signin-email\"\n className=\"githat-input\"\n type=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n placeholder=\"you@example.com\"\n autoComplete=\"email\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signin-password\">Password</label>\n <input\n id=\"githat-signin-password\"\n className=\"githat-input\"\n type=\"password\"\n value={password}\n onChange={(e) => setPassword(e.target.value)}\n placeholder=\"Enter your password\"\n autoComplete=\"current-password\"\n required\n />\n </div>\n {forgotPasswordUrl && (\n <a href={forgotPasswordUrl} className=\"githat-link githat-forgot-link\">Forgot password?</a>\n )}\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={loading || !email || !password || (email.length > 0 && !emailValid)}\n >\n {loading ? 'Signing in...' : 'Sign in'}\n </button>\n </form>\n {signUpUrl && (\n <p className=\"githat-form-footer\">\n Don't have an account? <a href={signUpUrl} className=\"githat-link\">Sign up</a>\n </p>\n )}\n <p className=\"githat-powered-by\">Secured by <strong>GitHat</strong></p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useAuth } from '../hooks';\n\ninterface SignUpFormProps {\n onSuccess?: (result: { requiresVerification: boolean; email: string }) => void;\n signInUrl?: string;\n}\n\nexport function SignUpForm({ onSuccess, signInUrl }: SignUpFormProps) {\n const { signUp, config } = useAuth();\n const [name, setName] = useState('');\n const [email, setEmail] = useState('');\n const [password, setPassword] = useState('');\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n const emailValid = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(email);\n const passwordValid = password.length >= 8\n && /[A-Z]/.test(password)\n && /[a-z]/.test(password)\n && /\\d/.test(password);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (!emailValid) {\n setError('Please enter a valid email address');\n return;\n }\n if (!passwordValid) {\n setError('Password must be 8+ characters with uppercase, lowercase, number, and special character');\n return;\n }\n setError('');\n setLoading(true);\n try {\n const result = await signUp({ email, password, name });\n if (onSuccess) {\n onSuccess(result);\n } else if (typeof window !== 'undefined') {\n window.location.href = config.afterSignInUrl!;\n }\n } catch (err: any) {\n setError(err.message || 'Sign up failed');\n } finally {\n setLoading(false);\n }\n };\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Create an account</h2>\n <p className=\"githat-form-subtitle\">Get started with GitHat</p>\n </div>\n {error && <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">{error}</div>}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Sign up form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-name\">Full name</label>\n <input\n id=\"githat-signup-name\"\n className=\"githat-input\"\n type=\"text\"\n value={name}\n onChange={(e) => setName(e.target.value)}\n placeholder=\"Your name\"\n autoComplete=\"name\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-email\">Email</label>\n <input\n id=\"githat-signup-email\"\n className=\"githat-input\"\n type=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n placeholder=\"you@example.com\"\n autoComplete=\"email\"\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-signup-password\">Password</label>\n <input\n id=\"githat-signup-password\"\n className=\"githat-input\"\n type=\"password\"\n value={password}\n onChange={(e) => setPassword(e.target.value)}\n placeholder=\"8+ characters\"\n autoComplete=\"new-password\"\n required\n />\n {password && !passwordValid && (\n <p className=\"githat-field-error\">\n Must be 8+ characters with uppercase, lowercase, and number\n </p>\n )}\n </div>\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={loading || !email || !password || !name || !emailValid}\n >\n {loading ? 'Creating account...' : 'Sign up'}\n </button>\n </form>\n {signInUrl && (\n <p className=\"githat-form-footer\">\n Already have an account? <a href={signInUrl} className=\"githat-link\">Sign in</a>\n </p>\n )}\n <p className=\"githat-powered-by\">Secured by <strong>GitHat</strong></p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useContext } from 'react';\nimport { GitHatContext } from '../provider';\n\ninterface SignInButtonProps {\n className?: string;\n children?: React.ReactNode;\n href?: string;\n}\n\nexport function SignInButton({ className, children, href }: SignInButtonProps) {\n const ctx = useContext(GitHatContext);\n const url = href || ctx?.config.signInUrl || '/sign-in';\n return (\n <a href={url} className={className || 'githat-button githat-button-primary'} aria-label=\"Sign in\">\n {children || 'Sign in'}\n </a>\n );\n}\n","'use client';\n\nimport React, { useContext } from 'react';\nimport { GitHatContext } from '../provider';\n\ninterface SignUpButtonProps {\n className?: string;\n children?: React.ReactNode;\n href?: string;\n}\n\nexport function SignUpButton({ className, children, href }: SignUpButtonProps) {\n const ctx = useContext(GitHatContext);\n const url = href || ctx?.config.signUpUrl || '/sign-up';\n return (\n <a href={url} className={className || 'githat-button githat-button-outline'} aria-label=\"Sign up\">\n {children || 'Sign up'}\n </a>\n );\n}\n","'use client';\n\nimport React, { useState, useRef, useEffect } from 'react';\nimport { useAuth } from '../hooks';\n\nexport function UserButton() {\n const { user, org, isSignedIn, signOut } = useAuth();\n const [open, setOpen] = useState(false);\n const ref = useRef<HTMLDivElement>(null);\n\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (ref.current && !ref.current.contains(e.target as Node)) setOpen(false);\n };\n document.addEventListener('mousedown', handleClickOutside);\n return () => document.removeEventListener('mousedown', handleClickOutside);\n }, []);\n\n if (!isSignedIn || !user) return null;\n\n const initials = user.name\n ? user.name.split(' ').map(n => n[0]).join('').toUpperCase().slice(0, 2)\n : user.email[0].toUpperCase();\n\n return (\n <div className=\"githat-user-button\" ref={ref}>\n <button className=\"githat-avatar-trigger\" onClick={() => setOpen(!open)} aria-label=\"User menu\" aria-expanded={open} aria-haspopup=\"true\">\n {user.avatarUrl ? (\n <img src={user.avatarUrl} alt={user.name || 'User avatar'} className=\"githat-avatar-img\" />\n ) : (\n <span className=\"githat-avatar-initials\">{initials}</span>\n )}\n </button>\n {open && (\n <div className=\"githat-dropdown\" role=\"menu\">\n <div className=\"githat-dropdown-header\">\n <p className=\"githat-dropdown-name\">{user.name}</p>\n <p className=\"githat-dropdown-email\">{user.email}</p>\n {org && <p className=\"githat-dropdown-org\">{org.name}</p>}\n </div>\n <div className=\"githat-dropdown-divider\" />\n <button className=\"githat-dropdown-item\" role=\"menuitem\" onClick={() => { signOut(); setOpen(false); }}>\n Sign out\n </button>\n </div>\n )}\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, useEffect, useRef } from 'react';\nimport { useAuth, useGitHat } from '../hooks';\nimport type { GitHatOrg } from '../types';\n\nexport function OrgSwitcher() {\n const { org, isSignedIn, switchOrg } = useAuth();\n const githat = useGitHat();\n const [orgs, setOrgs] = useState<GitHatOrg[]>([]);\n const [orgsLoading, setOrgsLoading] = useState(false);\n const [open, setOpen] = useState(false);\n const ref = useRef<HTMLDivElement>(null);\n\n useEffect(() => {\n if (isSignedIn) {\n setOrgsLoading(true);\n githat.getUserOrgs()\n .then(data => setOrgs(data.orgs || []))\n .catch(() => {})\n .finally(() => setOrgsLoading(false));\n }\n }, [isSignedIn]);\n\n useEffect(() => {\n const handleClickOutside = (e: MouseEvent) => {\n if (ref.current && !ref.current.contains(e.target as Node)) setOpen(false);\n };\n document.addEventListener('mousedown', handleClickOutside);\n return () => document.removeEventListener('mousedown', handleClickOutside);\n }, []);\n\n if (!isSignedIn || !org || (orgs.length < 2 && !orgsLoading)) return null;\n\n return (\n <div className=\"githat-org-switcher\" ref={ref}>\n <button className=\"githat-org-trigger\" onClick={() => setOpen(!open)} aria-label=\"Switch organization\" aria-expanded={open} aria-haspopup=\"true\">\n <span className=\"githat-org-name\">{org.name}</span>\n <span className=\"githat-chevron\">{open ? '\\u25B2' : '\\u25BC'}</span>\n </button>\n {open && (\n <div className=\"githat-dropdown\" role=\"menu\">\n {orgsLoading ? (\n <div className=\"githat-dropdown-item\" aria-busy=\"true\">Loading...</div>\n ) : orgs.map(o => (\n <button\n key={o.id}\n className={`githat-dropdown-item ${o.id === org.id ? 'githat-dropdown-item-active' : ''}`}\n role=\"menuitem\"\n aria-current={o.id === org.id ? 'true' : undefined}\n onClick={() => { switchOrg(o.id); setOpen(false); }}\n >\n {o.name}\n {o.id === org.id && <span className=\"githat-check\">{'\\u2713'}</span>}\n </button>\n ))}\n </div>\n )}\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, useEffect, useRef } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface VerifiedBadgeProps {\n type: 'mcp' | 'agent';\n identifier: string;\n label?: string;\n}\n\nconst CACHE_TTL = 5 * 60 * 1000; // 5 minutes\nconst cache = new Map<string, { verified: boolean; ts: number }>();\n\nexport function VerifiedBadge({ type, identifier, label }: VerifiedBadgeProps) {\n const githat = useGitHat();\n const [verified, setVerified] = useState<boolean | null>(null);\n const mounted = useRef(true);\n\n useEffect(() => {\n mounted.current = true;\n const key = `${type}:${identifier}`;\n const cached = cache.get(key);\n if (cached && Date.now() - cached.ts < CACHE_TTL) {\n setVerified(cached.verified);\n return;\n }\n\n const verify = type === 'mcp' ? githat.verifyMCP : githat.verifyAgent;\n verify(identifier)\n .then(data => {\n if (mounted.current) {\n setVerified(data.verified);\n cache.set(key, { verified: data.verified, ts: Date.now() });\n }\n })\n .catch(() => { if (mounted.current) setVerified(false); });\n\n return () => { mounted.current = false; };\n }, [type, identifier]);\n\n if (verified === null) return null;\n\n return (\n <span className={`githat-badge ${verified ? 'githat-badge-verified' : 'githat-badge-unverified'}`}>\n {verified ? '\\u2713' : '\\u2717'} {label || (verified ? 'Verified' : 'Unverified')}\n </span>\n );\n}\n","'use client';\n\nimport React from 'react';\nimport { useAuth } from '../hooks';\n\ninterface ProtectedRouteProps {\n children: React.ReactNode;\n fallback?: React.ReactNode;\n}\n\nexport function ProtectedRoute({ children, fallback }: ProtectedRouteProps) {\n const { isSignedIn, isLoading, config } = useAuth();\n\n if (isLoading) {\n return <>{fallback || <div className=\"githat-loading\">Loading...</div>}</>;\n }\n\n if (!isSignedIn) {\n if (typeof window !== 'undefined') {\n window.location.href = config.signInUrl!;\n }\n return null;\n }\n\n return <>{children}</>;\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface ForgotPasswordFormProps {\n onSuccess?: (email: string) => void;\n onError?: (error: Error) => void;\n signInUrl?: string;\n}\n\nexport function ForgotPasswordForm({\n onSuccess,\n onError,\n signInUrl = '/sign-in',\n}: ForgotPasswordFormProps) {\n const { forgotPassword } = useGitHat();\n const [email, setEmail] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [sent, setSent] = useState(false);\n const [error, setError] = useState('');\n\n const emailValid = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/.test(email);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (!emailValid) {\n setError('Please enter a valid email address');\n return;\n }\n setIsLoading(true);\n setError('');\n try {\n await forgotPassword(email);\n setSent(true);\n onSuccess?.(email);\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Failed to send reset email';\n setError(message);\n onError?.(err instanceof Error ? err : new Error(message));\n } finally {\n setIsLoading(false);\n }\n };\n\n if (sent) {\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Check your email</h2>\n <p className=\"githat-form-subtitle\">\n We sent a password reset link to <strong>{email}</strong>\n </p>\n </div>\n <a href={signInUrl} className=\"githat-link\">\n Back to sign in\n </a>\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n }\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Forgot password</h2>\n <p className=\"githat-form-subtitle\">\n Enter your email and we'll send you a reset link\n </p>\n </div>\n {error && (\n <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">\n {error}\n </div>\n )}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Forgot password form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-forgot-email\">\n Email\n </label>\n <input\n id=\"githat-forgot-email\"\n className=\"githat-input\"\n type=\"email\"\n value={email}\n onChange={(e) => setEmail(e.target.value)}\n placeholder=\"you@example.com\"\n autoComplete=\"email\"\n disabled={isLoading}\n required\n />\n </div>\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={isLoading || !email || (email.length > 0 && !emailValid)}\n >\n {isLoading ? 'Sending...' : 'Send reset link'}\n </button>\n </form>\n <p className=\"githat-form-footer\">\n Remember your password? <a href={signInUrl} className=\"githat-link\">Sign in</a>\n </p>\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface ResetPasswordFormProps {\n token: string;\n onSuccess?: () => void;\n onError?: (error: Error) => void;\n signInUrl?: string;\n minPasswordLength?: number;\n}\n\nexport function ResetPasswordForm({\n token,\n onSuccess,\n onError,\n signInUrl = '/sign-in',\n minPasswordLength = 8,\n}: ResetPasswordFormProps) {\n const { resetPassword } = useGitHat();\n const [password, setPassword] = useState('');\n const [confirm, setConfirm] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [success, setSuccess] = useState(false);\n const [error, setError] = useState('');\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (password !== confirm) {\n setError('Passwords do not match');\n return;\n }\n if (password.length < minPasswordLength) {\n setError(`Password must be at least ${minPasswordLength} characters`);\n return;\n }\n setIsLoading(true);\n setError('');\n try {\n await resetPassword(token, password);\n setSuccess(true);\n onSuccess?.();\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Failed to reset password';\n setError(message);\n onError?.(err instanceof Error ? err : new Error(message));\n } finally {\n setIsLoading(false);\n }\n };\n\n if (success) {\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Password reset!</h2>\n <p className=\"githat-form-subtitle\">\n Your password has been successfully reset.\n </p>\n </div>\n <a href={signInUrl} className=\"githat-button githat-button-primary\" style={{ display: 'block', textAlign: 'center', textDecoration: 'none' }}>\n Sign in\n </a>\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n }\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Reset password</h2>\n <p className=\"githat-form-subtitle\">Enter your new password</p>\n </div>\n {error && (\n <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">\n {error}\n </div>\n )}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Reset password form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-reset-password\">\n New password\n </label>\n <input\n id=\"githat-reset-password\"\n className=\"githat-input\"\n type=\"password\"\n value={password}\n onChange={(e) => setPassword(e.target.value)}\n placeholder=\"Enter new password\"\n autoComplete=\"new-password\"\n disabled={isLoading}\n required\n minLength={minPasswordLength}\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-reset-confirm\">\n Confirm password\n </label>\n <input\n id=\"githat-reset-confirm\"\n className=\"githat-input\"\n type=\"password\"\n value={confirm}\n onChange={(e) => setConfirm(e.target.value)}\n placeholder=\"Confirm new password\"\n autoComplete=\"new-password\"\n disabled={isLoading}\n required\n />\n </div>\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={isLoading || !password || !confirm}\n >\n {isLoading ? 'Resetting...' : 'Reset password'}\n </button>\n </form>\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useEffect, useState } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface VerifyEmailStatusProps {\n token: string;\n onSuccess?: () => void;\n onError?: (error: Error) => void;\n signInUrl?: string;\n redirectDelay?: number;\n}\n\nexport function VerifyEmailStatus({\n token,\n onSuccess,\n onError,\n signInUrl = '/sign-in',\n redirectDelay = 3000,\n}: VerifyEmailStatusProps) {\n const { verifyEmail } = useGitHat();\n const [status, setStatus] = useState<'loading' | 'success' | 'error'>('loading');\n const [error, setError] = useState('');\n\n useEffect(() => {\n if (!token) {\n setStatus('error');\n setError('Missing verification token');\n return;\n }\n\n verifyEmail(token)\n .then(() => {\n setStatus('success');\n onSuccess?.();\n if (signInUrl && redirectDelay > 0) {\n setTimeout(() => {\n window.location.href = signInUrl;\n }, redirectDelay);\n }\n })\n .catch((err) => {\n setStatus('error');\n const message = err instanceof Error ? err.message : 'Verification failed';\n setError(message);\n onError?.(err instanceof Error ? err : new Error(message));\n });\n }, [token, verifyEmail, onSuccess, onError, signInUrl, redirectDelay]);\n\n return (\n <div className=\"githat-form-container\">\n {status === 'loading' && (\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Verifying email...</h2>\n <p className=\"githat-form-subtitle\">Please wait while we verify your email address.</p>\n </div>\n )}\n {status === 'success' && (\n <>\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Email verified!</h2>\n <p className=\"githat-form-subtitle\">\n Your email has been successfully verified. Redirecting to sign in...\n </p>\n </div>\n <a href={signInUrl} className=\"githat-button githat-button-primary\" style={{ display: 'block', textAlign: 'center', textDecoration: 'none' }}>\n Sign in now\n </a>\n </>\n )}\n {status === 'error' && (\n <>\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Verification failed</h2>\n <p className=\"githat-form-subtitle\">{error}</p>\n </div>\n <p className=\"githat-form-footer\">\n The link may have expired. <a href=\"/sign-up\" className=\"githat-link\">Try signing up again</a>\n </p>\n </>\n )}\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n}\n","'use client';\n\nimport React, { useState, FormEvent } from 'react';\nimport { useGitHat } from '../hooks';\n\ninterface ChangePasswordFormProps {\n onSuccess?: () => void;\n onError?: (error: Error) => void;\n minPasswordLength?: number;\n}\n\nexport function ChangePasswordForm({\n onSuccess,\n onError,\n minPasswordLength = 8,\n}: ChangePasswordFormProps) {\n const { changePassword } = useGitHat();\n const [current, setCurrent] = useState('');\n const [newPass, setNewPass] = useState('');\n const [confirm, setConfirm] = useState('');\n const [isLoading, setIsLoading] = useState(false);\n const [error, setError] = useState('');\n const [success, setSuccess] = useState(false);\n\n const handleSubmit = async (e: FormEvent) => {\n e.preventDefault();\n if (newPass !== confirm) {\n setError('Passwords do not match');\n return;\n }\n if (newPass.length < minPasswordLength) {\n setError(`Password must be at least ${minPasswordLength} characters`);\n return;\n }\n setIsLoading(true);\n setError('');\n try {\n await changePassword(current, newPass);\n setSuccess(true);\n setCurrent('');\n setNewPass('');\n setConfirm('');\n onSuccess?.();\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Failed to change password';\n setError(message);\n onError?.(err instanceof Error ? err : new Error(message));\n } finally {\n setIsLoading(false);\n }\n };\n\n return (\n <div className=\"githat-form-container\">\n <div className=\"githat-form-header\">\n <h2 className=\"githat-form-title\">Change password</h2>\n <p className=\"githat-form-subtitle\">Update your account password</p>\n </div>\n {success && (\n <div className=\"githat-alert githat-alert-success\" role=\"status\" aria-live=\"polite\">\n Password changed successfully!\n </div>\n )}\n {error && (\n <div className=\"githat-alert githat-alert-error\" role=\"alert\" aria-live=\"polite\">\n {error}\n </div>\n )}\n <form onSubmit={handleSubmit} className=\"githat-form\" aria-label=\"Change password form\">\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-change-current\">\n Current password\n </label>\n <input\n id=\"githat-change-current\"\n className=\"githat-input\"\n type=\"password\"\n value={current}\n onChange={(e) => setCurrent(e.target.value)}\n placeholder=\"Enter current password\"\n autoComplete=\"current-password\"\n disabled={isLoading}\n required\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-change-new\">\n New password\n </label>\n <input\n id=\"githat-change-new\"\n className=\"githat-input\"\n type=\"password\"\n value={newPass}\n onChange={(e) => setNewPass(e.target.value)}\n placeholder=\"Enter new password\"\n autoComplete=\"new-password\"\n disabled={isLoading}\n required\n minLength={minPasswordLength}\n />\n </div>\n <div className=\"githat-field\">\n <label className=\"githat-label\" htmlFor=\"githat-change-confirm\">\n Confirm new password\n </label>\n <input\n id=\"githat-change-confirm\"\n className=\"githat-input\"\n type=\"password\"\n value={confirm}\n onChange={(e) => setConfirm(e.target.value)}\n placeholder=\"Confirm new password\"\n autoComplete=\"new-password\"\n disabled={isLoading}\n required\n />\n </div>\n <button\n type=\"submit\"\n className=\"githat-button githat-button-primary\"\n disabled={isLoading || !current || !newPass || !confirm}\n >\n {isLoading ? 'Changing...' : 'Change password'}\n </button>\n </form>\n <p className=\"githat-powered-by\">\n Secured by <strong>GitHat</strong>\n </p>\n </div>\n );\n}\n"],"mappings":"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","names":["useMemo","useCallback","useMemo","useCallback","useMemo","useMemo","useState","jsx","useState","useState","jsx","jsxs","useState","useContext","jsx","useContext","useContext","jsx","useContext","useState","useRef","useEffect","jsx","jsxs","useState","useRef","useEffect","useState","useEffect","useRef","jsx","jsxs","useState","useRef","useEffect","useState","useEffect","useRef","jsxs","useState","useRef","useEffect","jsx","useState","jsx","jsxs","useState","useState","jsx","jsxs","useState","useEffect","useState","Fragment","jsx","jsxs","useState","useEffect","useState","jsx","jsxs","useState"]}
package/dist/server.d.mts CHANGED
@@ -111,5 +111,54 @@ declare function updateOrgMetadata(orgId: string, metadata: OrgMetadata, options
111
111
  token: string;
112
112
  apiUrl?: string;
113
113
  }): Promise<OrgMetadata>;
114
+ /**
115
+ * Handler function that receives the request and verified auth payload.
116
+ */
117
+ type AuthenticatedHandler = (request: Request, auth: AuthPayload) => Promise<Response> | Response;
118
+ /**
119
+ * Options for the withAuth wrapper.
120
+ */
121
+ interface WithAuthOptions {
122
+ /**
123
+ * Secret key for local JWT verification.
124
+ * Defaults to process.env.GITHAT_SECRET_KEY if not provided.
125
+ */
126
+ secretKey?: string;
127
+ /**
128
+ * Custom response to return when authentication fails.
129
+ * Defaults to JSON { error: 'Unauthorized' } with status 401.
130
+ */
131
+ onUnauthorized?: () => Response;
132
+ }
133
+ /**
134
+ * Wrap an API route handler with authentication.
135
+ * The handler will only be called if the request has a valid auth token.
136
+ *
137
+ * @example
138
+ * ```ts
139
+ * // app/api/orders/route.ts
140
+ * import { withAuth } from '@githat/nextjs/server';
141
+ *
142
+ * export const GET = withAuth(async (request, auth) => {
143
+ * // auth.userId, auth.orgId, auth.role available
144
+ * const orders = await db.orders.findMany({ where: { orgId: auth.orgId } });
145
+ * return Response.json({ orders });
146
+ * }, { secretKey: process.env.GITHAT_SECRET_KEY });
147
+ * ```
148
+ *
149
+ * @example With custom unauthorized response
150
+ * ```ts
151
+ * export const GET = withAuth(
152
+ * async (request, auth) => {
153
+ * return Response.json({ userId: auth.userId });
154
+ * },
155
+ * {
156
+ * secretKey: process.env.GITHAT_SECRET_KEY,
157
+ * onUnauthorized: () => Response.redirect('/sign-in'),
158
+ * }
159
+ * );
160
+ * ```
161
+ */
162
+ declare function withAuth(handler: AuthenticatedHandler, options?: WithAuthOptions): (request: Request) => Promise<Response>;
114
163
 
115
- export { type AuthPayload, COOKIE_NAMES, type OrgMetadata, type VerifyOptions, getAuth, getOrgMetadata, updateOrgMetadata, verifyToken };
164
+ export { type AuthPayload, type AuthenticatedHandler, COOKIE_NAMES, type OrgMetadata, type VerifyOptions, type WithAuthOptions, getAuth, getOrgMetadata, updateOrgMetadata, verifyToken, withAuth };
package/dist/server.d.ts CHANGED
@@ -111,5 +111,54 @@ declare function updateOrgMetadata(orgId: string, metadata: OrgMetadata, options
111
111
  token: string;
112
112
  apiUrl?: string;
113
113
  }): Promise<OrgMetadata>;
114
+ /**
115
+ * Handler function that receives the request and verified auth payload.
116
+ */
117
+ type AuthenticatedHandler = (request: Request, auth: AuthPayload) => Promise<Response> | Response;
118
+ /**
119
+ * Options for the withAuth wrapper.
120
+ */
121
+ interface WithAuthOptions {
122
+ /**
123
+ * Secret key for local JWT verification.
124
+ * Defaults to process.env.GITHAT_SECRET_KEY if not provided.
125
+ */
126
+ secretKey?: string;
127
+ /**
128
+ * Custom response to return when authentication fails.
129
+ * Defaults to JSON { error: 'Unauthorized' } with status 401.
130
+ */
131
+ onUnauthorized?: () => Response;
132
+ }
133
+ /**
134
+ * Wrap an API route handler with authentication.
135
+ * The handler will only be called if the request has a valid auth token.
136
+ *
137
+ * @example
138
+ * ```ts
139
+ * // app/api/orders/route.ts
140
+ * import { withAuth } from '@githat/nextjs/server';
141
+ *
142
+ * export const GET = withAuth(async (request, auth) => {
143
+ * // auth.userId, auth.orgId, auth.role available
144
+ * const orders = await db.orders.findMany({ where: { orgId: auth.orgId } });
145
+ * return Response.json({ orders });
146
+ * }, { secretKey: process.env.GITHAT_SECRET_KEY });
147
+ * ```
148
+ *
149
+ * @example With custom unauthorized response
150
+ * ```ts
151
+ * export const GET = withAuth(
152
+ * async (request, auth) => {
153
+ * return Response.json({ userId: auth.userId });
154
+ * },
155
+ * {
156
+ * secretKey: process.env.GITHAT_SECRET_KEY,
157
+ * onUnauthorized: () => Response.redirect('/sign-in'),
158
+ * }
159
+ * );
160
+ * ```
161
+ */
162
+ declare function withAuth(handler: AuthenticatedHandler, options?: WithAuthOptions): (request: Request) => Promise<Response>;
114
163
 
115
- export { type AuthPayload, COOKIE_NAMES, type OrgMetadata, type VerifyOptions, getAuth, getOrgMetadata, updateOrgMetadata, verifyToken };
164
+ export { type AuthPayload, type AuthenticatedHandler, COOKIE_NAMES, type OrgMetadata, type VerifyOptions, type WithAuthOptions, getAuth, getOrgMetadata, updateOrgMetadata, verifyToken, withAuth };
package/dist/server.js CHANGED
@@ -34,7 +34,8 @@ __export(server_exports, {
34
34
  getAuth: () => getAuth,
35
35
  getOrgMetadata: () => getOrgMetadata,
36
36
  updateOrgMetadata: () => updateOrgMetadata,
37
- verifyToken: () => verifyToken
37
+ verifyToken: () => verifyToken,
38
+ withAuth: () => withAuth
38
39
  });
39
40
  module.exports = __toCommonJS(server_exports);
40
41
  var jose = __toESM(require("jose"));
@@ -175,12 +176,28 @@ async function updateOrgMetadata(orgId, metadata, options) {
175
176
  const data = await response.json();
176
177
  return data.metadata || {};
177
178
  }
179
+ function withAuth(handler, options = {}) {
180
+ return async (request) => {
181
+ const auth = await getAuth(request, { secretKey: options.secretKey });
182
+ if (!auth) {
183
+ if (options.onUnauthorized) {
184
+ return options.onUnauthorized();
185
+ }
186
+ return new Response(JSON.stringify({ error: "Unauthorized" }), {
187
+ status: 401,
188
+ headers: { "Content-Type": "application/json" }
189
+ });
190
+ }
191
+ return handler(request, auth);
192
+ };
193
+ }
178
194
  // Annotate the CommonJS export names for ESM import in node:
179
195
  0 && (module.exports = {
180
196
  COOKIE_NAMES,
181
197
  getAuth,
182
198
  getOrgMetadata,
183
199
  updateOrgMetadata,
184
- verifyToken
200
+ verifyToken,
201
+ withAuth
185
202
  });
186
203
  //# sourceMappingURL=server.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/server.ts","../src/config.ts"],"sourcesContent":["/**\n * @githat/nextjs/server\n *\n * Server-side utilities for token verification in Next.js API routes and middleware.\n * This module runs on the server only — do not import in client components.\n */\n\nimport * as jose from 'jose';\nimport { DEFAULT_API_URL } from './config';\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface AuthPayload {\n userId: string;\n email: string;\n orgId: string | null;\n orgSlug: string | null;\n role: 'owner' | 'admin' | 'member' | null;\n tier: 'free' | 'basic' | 'pro' | 'enterprise' | null;\n}\n\nexport interface VerifyOptions {\n /**\n * Secret key for local JWT verification. If provided, tokens are verified\n * locally without making an API call (~1ms vs ~50-100ms).\n * Must match the JWT_SECRET used by the GitHat backend.\n */\n secretKey?: string;\n\n /**\n * API URL for remote token verification. Defaults to https://api.githat.io\n */\n apiUrl?: string;\n}\n\nexport interface OrgMetadata {\n [key: string]: unknown;\n}\n\n// ============================================================================\n// Cookie Names\n// ============================================================================\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\n// ============================================================================\n// Token Verification\n// ============================================================================\n\n/**\n * Verify a JWT token and return the decoded auth payload.\n *\n * Supports two verification modes:\n * 1. Local verification (recommended for performance): Pass `secretKey` option\n * 2. API-based verification: No secret key needed, calls api.githat.io/auth/verify\n *\n * @example Local verification (fast, ~1ms)\n * ```ts\n * const auth = await verifyToken(token, {\n * secretKey: process.env.GITHAT_SECRET_KEY\n * });\n * ```\n *\n * @example API-based verification (simpler setup)\n * ```ts\n * const auth = await verifyToken(token);\n * ```\n */\nexport async function verifyToken(\n token: string,\n options: VerifyOptions = {}\n): Promise<AuthPayload> {\n const { secretKey, apiUrl = DEFAULT_API_URL } = options;\n\n if (secretKey) {\n // Local verification using jose\n return verifyTokenLocally(token, secretKey);\n }\n\n // API-based verification\n return verifyTokenViaAPI(token, apiUrl);\n}\n\n/**\n * Verify token locally using the secret key (no network call)\n */\nasync function verifyTokenLocally(\n token: string,\n secretKey: string\n): Promise<AuthPayload> {\n try {\n const secret = new TextEncoder().encode(secretKey);\n const { payload } = await jose.jwtVerify(token, secret, {\n algorithms: ['HS256'],\n });\n\n // Validate this is a user access token\n if (payload.type && payload.type !== 'user') {\n throw new Error('Invalid token type');\n }\n if (payload.tokenType !== 'access') {\n throw new Error('Invalid token type');\n }\n\n return {\n userId: payload.userId as string,\n email: payload.email as string,\n orgId: (payload.orgId as string) || null,\n orgSlug: (payload.orgSlug as string) || null,\n role: (payload.orgRole as AuthPayload['role']) || null,\n tier: null, // Tier requires DB lookup, not available in local verification\n };\n } catch (err) {\n if (err instanceof jose.errors.JWTExpired) {\n throw new Error('Token expired');\n }\n if (err instanceof jose.errors.JWTClaimValidationFailed) {\n throw new Error('Invalid token claims');\n }\n if (err instanceof jose.errors.JWTInvalid) {\n throw new Error('Invalid token');\n }\n // JWSSignatureVerificationFailed is used for invalid signatures\n if (err instanceof jose.errors.JWSSignatureVerificationFailed) {\n throw new Error('Invalid token signature');\n }\n throw err;\n }\n}\n\n/**\n * Verify token via API call to api.githat.io/auth/verify\n */\nasync function verifyTokenViaAPI(\n token: string,\n apiUrl: string\n): Promise<AuthPayload> {\n const response = await fetch(`${apiUrl}/auth/verify`, {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Token verification failed');\n }\n\n const data = await response.json();\n return {\n userId: data.userId,\n email: data.email,\n orgId: data.orgId,\n orgSlug: data.orgSlug,\n role: data.role,\n tier: data.tier,\n };\n}\n\n// ============================================================================\n// Request Helpers\n// ============================================================================\n\n/**\n * Extract and verify the auth token from a Next.js request.\n * Checks cookies first (for httpOnly cookie mode), then Authorization header.\n *\n * Returns null if no token is found (unauthenticated request).\n * Throws if a token is found but is invalid/expired.\n *\n * @example In a Next.js API route\n * ```ts\n * import { getAuth } from '@githat/nextjs/server';\n *\n * export async function GET(request: Request) {\n * const auth = await getAuth(request, {\n * secretKey: process.env.GITHAT_SECRET_KEY\n * });\n *\n * if (!auth) {\n * return Response.json({ error: 'Unauthorized' }, { status: 401 });\n * }\n *\n * return Response.json({ userId: auth.userId });\n * }\n * ```\n */\nexport async function getAuth(\n request: Request,\n options: VerifyOptions = {}\n): Promise<AuthPayload | null> {\n // Try to get token from cookie first (httpOnly cookie mode)\n const cookieHeader = request.headers.get('cookie');\n let token: string | null = null;\n\n if (cookieHeader) {\n const cookies = parseCookies(cookieHeader);\n token = cookies[COOKIE_NAMES.accessToken] || null;\n }\n\n // Fallback to Authorization header\n if (!token) {\n const authHeader = request.headers.get('authorization');\n if (authHeader?.startsWith('Bearer ')) {\n token = authHeader.slice(7);\n }\n }\n\n // No token found — return null (unauthenticated)\n if (!token) {\n return null;\n }\n\n // Verify the token\n return verifyToken(token, options);\n}\n\n/**\n * Parse cookies from a cookie header string\n */\nfunction parseCookies(cookieHeader: string): Record<string, string> {\n const cookies: Record<string, string> = {};\n const pairs = cookieHeader.split(';');\n\n for (const pair of pairs) {\n const [name, ...rest] = pair.trim().split('=');\n if (name) {\n cookies[name] = rest.join('=');\n }\n }\n\n return cookies;\n}\n\n// ============================================================================\n// Org Metadata (Server-side)\n// ============================================================================\n\n/**\n * Get organization metadata from the server.\n *\n * @example\n * ```ts\n * import { getOrgMetadata } from '@githat/nextjs/server';\n *\n * const meta = await getOrgMetadata(orgId, {\n * token: accessToken,\n * apiUrl: 'https://api.githat.io'\n * });\n * console.log(meta.stripeAccountId);\n * ```\n */\nexport async function getOrgMetadata(\n orgId: string,\n options: { token: string; apiUrl?: string }\n): Promise<OrgMetadata> {\n const { token, apiUrl = DEFAULT_API_URL } = options;\n\n const response = await fetch(`${apiUrl}/orgs/${orgId}/metadata`, {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Failed to get org metadata');\n }\n\n const data = await response.json();\n return data.metadata || {};\n}\n\n/**\n * Update organization metadata from the server.\n *\n * @example\n * ```ts\n * import { updateOrgMetadata } from '@githat/nextjs/server';\n *\n * await updateOrgMetadata(orgId, { stripeAccountId: 'acct_xxx' }, {\n * token: accessToken,\n * apiUrl: 'https://api.githat.io'\n * });\n * ```\n */\nexport async function updateOrgMetadata(\n orgId: string,\n metadata: OrgMetadata,\n options: { token: string; apiUrl?: string }\n): Promise<OrgMetadata> {\n const { token, apiUrl = DEFAULT_API_URL } = options;\n\n const response = await fetch(`${apiUrl}/orgs/${orgId}/metadata`, {\n method: 'PATCH',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Failed to update org metadata');\n }\n\n const data = await response.json();\n return data.metadata || {};\n}\n","import type { GitHatConfig } from './types';\n\nexport const DEFAULT_API_URL = 'https://api.githat.io';\n\nexport const TOKEN_KEYS = {\n accessToken: 'githat_access_token',\n refreshToken: 'githat_refresh_token',\n user: 'githat_user',\n org: 'githat_org',\n} as const;\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\nexport function resolveConfig(config: GitHatConfig): Required<GitHatConfig> {\n return {\n publishableKey: config.publishableKey,\n apiUrl: config.apiUrl || DEFAULT_API_URL,\n signInUrl: config.signInUrl || '/sign-in',\n signUpUrl: config.signUpUrl || '/sign-up',\n afterSignInUrl: config.afterSignInUrl || '/dashboard',\n afterSignOutUrl: config.afterSignOutUrl || '/',\n tokenStorage: config.tokenStorage || 'localStorage',\n };\n}\n"],"mappings":"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","names":["data"]}
1
+ {"version":3,"sources":["../src/server.ts","../src/config.ts"],"sourcesContent":["/**\n * @githat/nextjs/server\n *\n * Server-side utilities for token verification in Next.js API routes and middleware.\n * This module runs on the server only — do not import in client components.\n */\n\nimport * as jose from 'jose';\nimport { DEFAULT_API_URL } from './config';\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface AuthPayload {\n userId: string;\n email: string;\n orgId: string | null;\n orgSlug: string | null;\n role: 'owner' | 'admin' | 'member' | null;\n tier: 'free' | 'basic' | 'pro' | 'enterprise' | null;\n}\n\nexport interface VerifyOptions {\n /**\n * Secret key for local JWT verification. If provided, tokens are verified\n * locally without making an API call (~1ms vs ~50-100ms).\n * Must match the JWT_SECRET used by the GitHat backend.\n */\n secretKey?: string;\n\n /**\n * API URL for remote token verification. Defaults to https://api.githat.io\n */\n apiUrl?: string;\n}\n\nexport interface OrgMetadata {\n [key: string]: unknown;\n}\n\n// ============================================================================\n// Cookie Names\n// ============================================================================\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\n// ============================================================================\n// Token Verification\n// ============================================================================\n\n/**\n * Verify a JWT token and return the decoded auth payload.\n *\n * Supports two verification modes:\n * 1. Local verification (recommended for performance): Pass `secretKey` option\n * 2. API-based verification: No secret key needed, calls api.githat.io/auth/verify\n *\n * @example Local verification (fast, ~1ms)\n * ```ts\n * const auth = await verifyToken(token, {\n * secretKey: process.env.GITHAT_SECRET_KEY\n * });\n * ```\n *\n * @example API-based verification (simpler setup)\n * ```ts\n * const auth = await verifyToken(token);\n * ```\n */\nexport async function verifyToken(\n token: string,\n options: VerifyOptions = {}\n): Promise<AuthPayload> {\n const { secretKey, apiUrl = DEFAULT_API_URL } = options;\n\n if (secretKey) {\n // Local verification using jose\n return verifyTokenLocally(token, secretKey);\n }\n\n // API-based verification\n return verifyTokenViaAPI(token, apiUrl);\n}\n\n/**\n * Verify token locally using the secret key (no network call)\n */\nasync function verifyTokenLocally(\n token: string,\n secretKey: string\n): Promise<AuthPayload> {\n try {\n const secret = new TextEncoder().encode(secretKey);\n const { payload } = await jose.jwtVerify(token, secret, {\n algorithms: ['HS256'],\n });\n\n // Validate this is a user access token\n if (payload.type && payload.type !== 'user') {\n throw new Error('Invalid token type');\n }\n if (payload.tokenType !== 'access') {\n throw new Error('Invalid token type');\n }\n\n return {\n userId: payload.userId as string,\n email: payload.email as string,\n orgId: (payload.orgId as string) || null,\n orgSlug: (payload.orgSlug as string) || null,\n role: (payload.orgRole as AuthPayload['role']) || null,\n tier: null, // Tier requires DB lookup, not available in local verification\n };\n } catch (err) {\n if (err instanceof jose.errors.JWTExpired) {\n throw new Error('Token expired');\n }\n if (err instanceof jose.errors.JWTClaimValidationFailed) {\n throw new Error('Invalid token claims');\n }\n if (err instanceof jose.errors.JWTInvalid) {\n throw new Error('Invalid token');\n }\n // JWSSignatureVerificationFailed is used for invalid signatures\n if (err instanceof jose.errors.JWSSignatureVerificationFailed) {\n throw new Error('Invalid token signature');\n }\n throw err;\n }\n}\n\n/**\n * Verify token via API call to api.githat.io/auth/verify\n */\nasync function verifyTokenViaAPI(\n token: string,\n apiUrl: string\n): Promise<AuthPayload> {\n const response = await fetch(`${apiUrl}/auth/verify`, {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Token verification failed');\n }\n\n const data = await response.json();\n return {\n userId: data.userId,\n email: data.email,\n orgId: data.orgId,\n orgSlug: data.orgSlug,\n role: data.role,\n tier: data.tier,\n };\n}\n\n// ============================================================================\n// Request Helpers\n// ============================================================================\n\n/**\n * Extract and verify the auth token from a Next.js request.\n * Checks cookies first (for httpOnly cookie mode), then Authorization header.\n *\n * Returns null if no token is found (unauthenticated request).\n * Throws if a token is found but is invalid/expired.\n *\n * @example In a Next.js API route\n * ```ts\n * import { getAuth } from '@githat/nextjs/server';\n *\n * export async function GET(request: Request) {\n * const auth = await getAuth(request, {\n * secretKey: process.env.GITHAT_SECRET_KEY\n * });\n *\n * if (!auth) {\n * return Response.json({ error: 'Unauthorized' }, { status: 401 });\n * }\n *\n * return Response.json({ userId: auth.userId });\n * }\n * ```\n */\nexport async function getAuth(\n request: Request,\n options: VerifyOptions = {}\n): Promise<AuthPayload | null> {\n // Try to get token from cookie first (httpOnly cookie mode)\n const cookieHeader = request.headers.get('cookie');\n let token: string | null = null;\n\n if (cookieHeader) {\n const cookies = parseCookies(cookieHeader);\n token = cookies[COOKIE_NAMES.accessToken] || null;\n }\n\n // Fallback to Authorization header\n if (!token) {\n const authHeader = request.headers.get('authorization');\n if (authHeader?.startsWith('Bearer ')) {\n token = authHeader.slice(7);\n }\n }\n\n // No token found — return null (unauthenticated)\n if (!token) {\n return null;\n }\n\n // Verify the token\n return verifyToken(token, options);\n}\n\n/**\n * Parse cookies from a cookie header string\n */\nfunction parseCookies(cookieHeader: string): Record<string, string> {\n const cookies: Record<string, string> = {};\n const pairs = cookieHeader.split(';');\n\n for (const pair of pairs) {\n const [name, ...rest] = pair.trim().split('=');\n if (name) {\n cookies[name] = rest.join('=');\n }\n }\n\n return cookies;\n}\n\n// ============================================================================\n// Org Metadata (Server-side)\n// ============================================================================\n\n/**\n * Get organization metadata from the server.\n *\n * @example\n * ```ts\n * import { getOrgMetadata } from '@githat/nextjs/server';\n *\n * const meta = await getOrgMetadata(orgId, {\n * token: accessToken,\n * apiUrl: 'https://api.githat.io'\n * });\n * console.log(meta.stripeAccountId);\n * ```\n */\nexport async function getOrgMetadata(\n orgId: string,\n options: { token: string; apiUrl?: string }\n): Promise<OrgMetadata> {\n const { token, apiUrl = DEFAULT_API_URL } = options;\n\n const response = await fetch(`${apiUrl}/orgs/${orgId}/metadata`, {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Failed to get org metadata');\n }\n\n const data = await response.json();\n return data.metadata || {};\n}\n\n/**\n * Update organization metadata from the server.\n *\n * @example\n * ```ts\n * import { updateOrgMetadata } from '@githat/nextjs/server';\n *\n * await updateOrgMetadata(orgId, { stripeAccountId: 'acct_xxx' }, {\n * token: accessToken,\n * apiUrl: 'https://api.githat.io'\n * });\n * ```\n */\nexport async function updateOrgMetadata(\n orgId: string,\n metadata: OrgMetadata,\n options: { token: string; apiUrl?: string }\n): Promise<OrgMetadata> {\n const { token, apiUrl = DEFAULT_API_URL } = options;\n\n const response = await fetch(`${apiUrl}/orgs/${orgId}/metadata`, {\n method: 'PATCH',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(metadata),\n });\n\n if (!response.ok) {\n const data = await response.json().catch(() => ({}));\n throw new Error(data.error || 'Failed to update org metadata');\n }\n\n const data = await response.json();\n return data.metadata || {};\n}\n\n// ============================================================================\n// API Route Helpers\n// ============================================================================\n\n/**\n * Handler function that receives the request and verified auth payload.\n */\nexport type AuthenticatedHandler = (\n request: Request,\n auth: AuthPayload\n) => Promise<Response> | Response;\n\n/**\n * Options for the withAuth wrapper.\n */\nexport interface WithAuthOptions {\n /**\n * Secret key for local JWT verification.\n * Defaults to process.env.GITHAT_SECRET_KEY if not provided.\n */\n secretKey?: string;\n\n /**\n * Custom response to return when authentication fails.\n * Defaults to JSON { error: 'Unauthorized' } with status 401.\n */\n onUnauthorized?: () => Response;\n}\n\n/**\n * Wrap an API route handler with authentication.\n * The handler will only be called if the request has a valid auth token.\n *\n * @example\n * ```ts\n * // app/api/orders/route.ts\n * import { withAuth } from '@githat/nextjs/server';\n *\n * export const GET = withAuth(async (request, auth) => {\n * // auth.userId, auth.orgId, auth.role available\n * const orders = await db.orders.findMany({ where: { orgId: auth.orgId } });\n * return Response.json({ orders });\n * }, { secretKey: process.env.GITHAT_SECRET_KEY });\n * ```\n *\n * @example With custom unauthorized response\n * ```ts\n * export const GET = withAuth(\n * async (request, auth) => {\n * return Response.json({ userId: auth.userId });\n * },\n * {\n * secretKey: process.env.GITHAT_SECRET_KEY,\n * onUnauthorized: () => Response.redirect('/sign-in'),\n * }\n * );\n * ```\n */\nexport function withAuth(\n handler: AuthenticatedHandler,\n options: WithAuthOptions = {}\n): (request: Request) => Promise<Response> {\n return async (request: Request): Promise<Response> => {\n const auth = await getAuth(request, { secretKey: options.secretKey });\n\n if (!auth) {\n if (options.onUnauthorized) {\n return options.onUnauthorized();\n }\n return new Response(JSON.stringify({ error: 'Unauthorized' }), {\n status: 401,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n return handler(request, auth);\n };\n}\n","import type { GitHatConfig } from './types';\n\nexport const DEFAULT_API_URL = 'https://api.githat.io';\n\nexport const TOKEN_KEYS = {\n accessToken: 'githat_access_token',\n refreshToken: 'githat_refresh_token',\n user: 'githat_user',\n org: 'githat_org',\n} as const;\n\nexport const COOKIE_NAMES = {\n accessToken: 'githat_access',\n refreshToken: 'githat_refresh',\n} as const;\n\nexport function resolveConfig(config: GitHatConfig): Required<GitHatConfig> {\n return {\n publishableKey: config.publishableKey,\n apiUrl: config.apiUrl || DEFAULT_API_URL,\n signInUrl: config.signInUrl || '/sign-in',\n signUpUrl: config.signUpUrl || '/sign-up',\n afterSignInUrl: config.afterSignInUrl || '/dashboard',\n afterSignOutUrl: config.afterSignOutUrl || '/',\n tokenStorage: config.tokenStorage || 'localStorage',\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAOA,WAAsB;;;ACLf,IAAM,kBAAkB;;;AD2CxB,IAAM,eAAe;AAAA,EAC1B,aAAa;AAAA,EACb,cAAc;AAChB;AAyBA,eAAsB,YACpB,OACA,UAAyB,CAAC,GACJ;AACtB,QAAM,EAAE,WAAW,SAAS,gBAAgB,IAAI;AAEhD,MAAI,WAAW;AAEb,WAAO,mBAAmB,OAAO,SAAS;AAAA,EAC5C;AAGA,SAAO,kBAAkB,OAAO,MAAM;AACxC;AAKA,eAAe,mBACb,OACA,WACsB;AACtB,MAAI;AACF,UAAM,SAAS,IAAI,YAAY,EAAE,OAAO,SAAS;AACjD,UAAM,EAAE,QAAQ,IAAI,MAAW,eAAU,OAAO,QAAQ;AAAA,MACtD,YAAY,CAAC,OAAO;AAAA,IACtB,CAAC;AAGD,QAAI,QAAQ,QAAQ,QAAQ,SAAS,QAAQ;AAC3C,YAAM,IAAI,MAAM,oBAAoB;AAAA,IACtC;AACA,QAAI,QAAQ,cAAc,UAAU;AAClC,YAAM,IAAI,MAAM,oBAAoB;AAAA,IACtC;AAEA,WAAO;AAAA,MACL,QAAQ,QAAQ;AAAA,MAChB,OAAO,QAAQ;AAAA,MACf,OAAQ,QAAQ,SAAoB;AAAA,MACpC,SAAU,QAAQ,WAAsB;AAAA,MACxC,MAAO,QAAQ,WAAmC;AAAA,MAClD,MAAM;AAAA;AAAA,IACR;AAAA,EACF,SAAS,KAAK;AACZ,QAAI,eAAoB,YAAO,YAAY;AACzC,YAAM,IAAI,MAAM,eAAe;AAAA,IACjC;AACA,QAAI,eAAoB,YAAO,0BAA0B;AACvD,YAAM,IAAI,MAAM,sBAAsB;AAAA,IACxC;AACA,QAAI,eAAoB,YAAO,YAAY;AACzC,YAAM,IAAI,MAAM,eAAe;AAAA,IACjC;AAEA,QAAI,eAAoB,YAAO,gCAAgC;AAC7D,YAAM,IAAI,MAAM,yBAAyB;AAAA,IAC3C;AACA,UAAM;AAAA,EACR;AACF;AAKA,eAAe,kBACb,OACA,QACsB;AACtB,QAAM,WAAW,MAAM,MAAM,GAAG,MAAM,gBAAgB;AAAA,IACpD,QAAQ;AAAA,IACR,SAAS;AAAA,MACP,eAAe,UAAU,KAAK;AAAA,MAC9B,gBAAgB;AAAA,IAClB;AAAA,EACF,CAAC;AAED,MAAI,CAAC,SAAS,IAAI;AAChB,UAAMA,QAAO,MAAM,SAAS,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACnD,UAAM,IAAI,MAAMA,MAAK,SAAS,2BAA2B;AAAA,EAC3D;AAEA,QAAM,OAAO,MAAM,SAAS,KAAK;AACjC,SAAO;AAAA,IACL,QAAQ,KAAK;AAAA,IACb,OAAO,KAAK;AAAA,IACZ,OAAO,KAAK;AAAA,IACZ,SAAS,KAAK;AAAA,IACd,MAAM,KAAK;AAAA,IACX,MAAM,KAAK;AAAA,EACb;AACF;AA8BA,eAAsB,QACpB,SACA,UAAyB,CAAC,GACG;AAE7B,QAAM,eAAe,QAAQ,QAAQ,IAAI,QAAQ;AACjD,MAAI,QAAuB;AAE3B,MAAI,cAAc;AAChB,UAAM,UAAU,aAAa,YAAY;AACzC,YAAQ,QAAQ,aAAa,WAAW,KAAK;AAAA,EAC/C;AAGA,MAAI,CAAC,OAAO;AACV,UAAM,aAAa,QAAQ,QAAQ,IAAI,eAAe;AACtD,QAAI,YAAY,WAAW,SAAS,GAAG;AACrC,cAAQ,WAAW,MAAM,CAAC;AAAA,IAC5B;AAAA,EACF;AAGA,MAAI,CAAC,OAAO;AACV,WAAO;AAAA,EACT;AAGA,SAAO,YAAY,OAAO,OAAO;AACnC;AAKA,SAAS,aAAa,cAA8C;AAClE,QAAM,UAAkC,CAAC;AACzC,QAAM,QAAQ,aAAa,MAAM,GAAG;AAEpC,aAAW,QAAQ,OAAO;AACxB,UAAM,CAAC,MAAM,GAAG,IAAI,IAAI,KAAK,KAAK,EAAE,MAAM,GAAG;AAC7C,QAAI,MAAM;AACR,cAAQ,IAAI,IAAI,KAAK,KAAK,GAAG;AAAA,IAC/B;AAAA,EACF;AAEA,SAAO;AACT;AAoBA,eAAsB,eACpB,OACA,SACsB;AACtB,QAAM,EAAE,OAAO,SAAS,gBAAgB,IAAI;AAE5C,QAAM,WAAW,MAAM,MAAM,GAAG,MAAM,SAAS,KAAK,aAAa;AAAA,IAC/D,QAAQ;AAAA,IACR,SAAS;AAAA,MACP,eAAe,UAAU,KAAK;AAAA,MAC9B,gBAAgB;AAAA,IAClB;AAAA,EACF,CAAC;AAED,MAAI,CAAC,SAAS,IAAI;AAChB,UAAMA,QAAO,MAAM,SAAS,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACnD,UAAM,IAAI,MAAMA,MAAK,SAAS,4BAA4B;AAAA,EAC5D;AAEA,QAAM,OAAO,MAAM,SAAS,KAAK;AACjC,SAAO,KAAK,YAAY,CAAC;AAC3B;AAeA,eAAsB,kBACpB,OACA,UACA,SACsB;AACtB,QAAM,EAAE,OAAO,SAAS,gBAAgB,IAAI;AAE5C,QAAM,WAAW,MAAM,MAAM,GAAG,MAAM,SAAS,KAAK,aAAa;AAAA,IAC/D,QAAQ;AAAA,IACR,SAAS;AAAA,MACP,eAAe,UAAU,KAAK;AAAA,MAC9B,gBAAgB;AAAA,IAClB;AAAA,IACA,MAAM,KAAK,UAAU,QAAQ;AAAA,EAC/B,CAAC;AAED,MAAI,CAAC,SAAS,IAAI;AAChB,UAAMA,QAAO,MAAM,SAAS,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACnD,UAAM,IAAI,MAAMA,MAAK,SAAS,+BAA+B;AAAA,EAC/D;AAEA,QAAM,OAAO,MAAM,SAAS,KAAK;AACjC,SAAO,KAAK,YAAY,CAAC;AAC3B;AA4DO,SAAS,SACd,SACA,UAA2B,CAAC,GACa;AACzC,SAAO,OAAO,YAAwC;AACpD,UAAM,OAAO,MAAM,QAAQ,SAAS,EAAE,WAAW,QAAQ,UAAU,CAAC;AAEpE,QAAI,CAAC,MAAM;AACT,UAAI,QAAQ,gBAAgB;AAC1B,eAAO,QAAQ,eAAe;AAAA,MAChC;AACA,aAAO,IAAI,SAAS,KAAK,UAAU,EAAE,OAAO,eAAe,CAAC,GAAG;AAAA,QAC7D,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,MAChD,CAAC;AAAA,IACH;AAEA,WAAO,QAAQ,SAAS,IAAI;AAAA,EAC9B;AACF;","names":["data"]}