ns-auth-sdk 1.3.0 → 1.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -53,6 +53,7 @@ const authService = new AuthService({
53
53
  rpId: 'your-domain.com',
54
54
  rpName: 'Your App Name',
55
55
  storageKey: 'nsauth_keyinfo',
56
+ cacheOnCreation: true, // Enable early caching (default: true)
56
57
  });
57
58
 
58
59
  // Initialize relay service with EventStore
@@ -182,6 +183,33 @@ function MembershipPageComponent() {
182
183
  - `clearStoredKeyInfo(): void` - Clear stored key info
183
184
  - `isPrfSupported(): Promise<boolean>` - Check if PRF is supported
184
185
 
186
+ #### Configuration Options
187
+
188
+ ```typescript
189
+ interface NosskeyManagerOptions {
190
+ cacheOptions?: {
191
+ enabled: boolean;
192
+ timeoutMs?: number;
193
+ cacheOnCreation?: boolean; // Cache key immediately after derivation (default: true)
194
+ };
195
+ storageOptions?: {
196
+ enabled: boolean;
197
+ storage?: Storage;
198
+ storageKey?: string;
199
+ };
200
+ prfOptions?: {
201
+ rpId?: string;
202
+ timeout?: number;
203
+ userVerification?: UserVerificationRequirement;
204
+ };
205
+ }
206
+ ```
207
+
208
+ **Cache Options:**
209
+ - `enabled`: Enable/disable key caching
210
+ - `timeoutMs`: Cache timeout in milliseconds (default: 30 minutes)
211
+ - `cacheOnCreation`: When `true`, caches the key immediately after `createNostrKey()` to reduce biometric prompts from 2-3 to 1-2. This is enabled by default for better user experience.
212
+
185
213
  ### RelayService
186
214
 
187
215
  Service for communicating with Nostr relays using applesauce-core.
package/dist/index.cjs CHANGED
@@ -25,7 +25,7 @@ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__ge
25
25
  }) : target, mod));
26
26
 
27
27
  //#endregion
28
- const require_utils = require('./utils-CxLeFzNn.cjs');
28
+ const require_utils = require('./utils-BUM1trwg.cjs');
29
29
  let react = require("react");
30
30
  let react_jsx_runtime = require("react/jsx-runtime");
31
31
  let isomorphic_dompurify = require("isomorphic-dompurify");
@@ -45,7 +45,8 @@ var AuthService = class {
45
45
  rpId: config.rpId || (typeof window !== "undefined" ? window.location.hostname.replace(/^www\./, "") : "localhost"),
46
46
  rpName: config.rpName || this.getDefaultRpName(),
47
47
  storageKey: config.storageKey || "nsauth_keyinfo",
48
- cacheTimeoutMs: config.cacheTimeoutMs || 1800 * 1e3
48
+ cacheTimeoutMs: config.cacheTimeoutMs || 1800 * 1e3,
49
+ cacheOnCreation: config.cacheOnCreation !== void 0 ? config.cacheOnCreation : true
49
50
  };
50
51
  }
51
52
  getDefaultRpName() {
@@ -61,7 +62,8 @@ var AuthService = class {
61
62
  if (!this.manager) this.manager = new require_utils.NosskeyManager({
62
63
  cacheOptions: {
63
64
  enabled: true,
64
- timeoutMs: this.config.cacheTimeoutMs
65
+ timeoutMs: this.config.cacheTimeoutMs,
66
+ cacheOnCreation: this.config.cacheOnCreation
65
67
  },
66
68
  storageOptions: {
67
69
  enabled: true,
@@ -144,7 +146,7 @@ var AuthService = class {
144
146
  * Check if PRF is supported
145
147
  */
146
148
  async isPrfSupported() {
147
- const { isPrfSupported } = await Promise.resolve().then(() => require("./utils-BR2TqhEA.cjs"));
149
+ const { isPrfSupported } = await Promise.resolve().then(() => require("./utils-BV-7ST9t.cjs"));
148
150
  return await isPrfSupported();
149
151
  }
150
152
  };
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","names":["NosskeyManager","DOMPurify"],"sources":["../src/services/auth.service.ts","../src/utils/validation.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/utils/sanitize.ts","../src/components/membership/BarcodeScanner.tsx","../src/components/membership/MembershipPage.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":["import { NosskeyManager, type NostrKeyInfo, type NostrEvent, type PasskeyCreationOptions } from '../utils';\nimport type { AuthServiceConfig } from '../types/auth';\n\n/**\n * Service wrapper around NosskeyManager\n * Handles WebAuthn/Passkey integration with Nostr\n */\nexport class AuthService {\n private manager: NosskeyManager | null = null;\n private config: Required<AuthServiceConfig>;\n\n constructor(config: AuthServiceConfig = {}) {\n this.config = {\n rpId: config.rpId || (typeof window !== 'undefined' ? window.location.hostname.replace(/^www\\./, '') : 'localhost'),\n rpName: config.rpName || this.getDefaultRpName(),\n storageKey: config.storageKey || 'nsauth_keyinfo',\n cacheTimeoutMs: config.cacheTimeoutMs || 30 * 60 * 1000,\n };\n }\n\n private getDefaultRpName(): string {\n if (typeof window === 'undefined') return 'localhost';\n const hostname = window.location.hostname;\n if (hostname.includes('nosskey.app')) return 'nosskey.app';\n return hostname.replace(/^www\\./, '');\n }\n\n /**\n * Initialize the NosskeyManager instance\n */\n private getManager(): NosskeyManager {\n if (!this.manager) {\n this.manager = new NosskeyManager({\n cacheOptions: {\n enabled: true,\n timeoutMs: this.config.cacheTimeoutMs,\n },\n storageOptions: {\n enabled: true,\n storageKey: this.config.storageKey,\n },\n });\n }\n return this.manager;\n }\n\n /**\n * Create a new passkey\n * Uses platform authenticator only (Touch ID, Face ID, Windows Hello)\n */\n async createPasskey(username?: string): Promise<Uint8Array> {\n const manager = this.getManager();\n \n const rpId = this.config.rpId === 'localhost' ? 'localhost' : this.config.rpId;\n const rpName = this.config.rpName;\n \n const trimmedUsername = username?.trim();\n const uniqueUsername = trimmedUsername \n ? trimmedUsername \n : `user-${Date.now()}@example.com`;\n \n const options: PasskeyCreationOptions = {\n rp: {\n id: rpId,\n name: rpName,\n },\n user: {\n name: uniqueUsername,\n displayName: trimmedUsername || 'User',\n },\n authenticatorSelection: {\n authenticatorAttachment: 'platform',\n residentKey: 'preferred',\n userVerification: 'preferred',\n },\n extensions: {\n prf: {},\n },\n };\n \n return await manager.createPasskey(options);\n }\n\n /**\n * Create a new Nostr key from a credential ID\n */\n async createNostrKey(credentialId?: Uint8Array): Promise<NostrKeyInfo> {\n const manager = this.getManager();\n return await manager.createNostrKey(credentialId);\n }\n\n /**\n * Get the current public key\n */\n async getPublicKey(): Promise<string> {\n const manager = this.getManager();\n return await manager.getPublicKey();\n }\n\n /**\n * Sign a Nostr event\n */\n async signEvent(event: NostrEvent): Promise<NostrEvent> {\n const manager = this.getManager();\n return await manager.signEvent(event);\n }\n\n /**\n * Get current key info\n */\n getCurrentKeyInfo(): NostrKeyInfo | null {\n const manager = this.getManager();\n return manager.getCurrentKeyInfo();\n }\n\n /**\n * Set current key info\n */\n setCurrentKeyInfo(keyInfo: NostrKeyInfo): void {\n const manager = this.getManager();\n manager.setCurrentKeyInfo(keyInfo);\n }\n\n /**\n * Check if key info exists\n */\n hasKeyInfo(): boolean {\n const manager = this.getManager();\n return manager.hasKeyInfo();\n }\n\n /**\n * Clear stored key info\n */\n clearStoredKeyInfo(): void {\n const manager = this.getManager();\n manager.clearStoredKeyInfo();\n }\n\n /**\n * Check if PRF is supported\n */\n async isPrfSupported(): Promise<boolean> {\n const { isPrfSupported } = await import('../utils');\n return await isPrfSupported();\n }\n}\n\n","export const MAX_ROLE_LENGTH = 100;\nconst ROLE_PATTERN = /^[a-zA-Z0-9\\s\\-_]+$/;\n\nexport const isValidRoleTag = (role: string): boolean => {\n const trimmed = role.trim();\n return (\n trimmed.length > 0 &&\n trimmed.length <= MAX_ROLE_LENGTH &&\n ROLE_PATTERN.test(trimmed)\n );\n};\n\nexport const isValidPubkey = (pubkey: string): boolean =>\n pubkey.length === 64 && /^[0-9a-fA-F]+$/.test(pubkey);\n\nexport const isValidHttpUrl = (url: string): boolean => {\n try {\n const parsed = new URL(url);\n return ['http:', 'https:'].includes(parsed.protocol);\n } catch {\n return false;\n }\n};\n\nexport const isValidRelayUrl = (url: string): boolean => {\n try {\n const parsed = new URL(url);\n return ['ws:', 'wss:'].includes(parsed.protocol) && parsed.hostname.length > 0;\n } catch {\n return false;\n }\n};\n","import type { EventStore } from 'applesauce-core';\nimport type { NostrEvent } from '../types/nostr';\nimport type { ProfileMetadata, FollowEntry } from '../types/nostr';\nimport type { RelayServiceConfig } from '../types/auth';\nimport { isValidPubkey, isValidRelayUrl, isValidRoleTag } from '../utils/validation';\n\n/**\n * Service for communicating with Nostr relays using applesauce-core\n */\nexport class RelayService {\n private eventStore: EventStore | null = null;\n private relayUrls: string[];\n private defaultRelays = ['wss://relay.damus.io'];\n private readonly maxProfileContentSize = 10000;\n private readonly minProfileQueryIntervalMs = 300;\n private readonly minPublishIntervalMs = 750;\n private readonly lastActionAt = new Map<string, number>();\n\n constructor(config: RelayServiceConfig = {}) {\n this.relayUrls = this.validateRelayUrls(config.relayUrls ?? this.defaultRelays);\n }\n\n /**\n * Initialize with applesauce EventStore\n */\n initialize(eventStore: EventStore): void {\n this.eventStore = eventStore;\n // Set default relays if EventStore has a method for it\n if (eventStore && 'setRelays' in eventStore && typeof eventStore.setRelays === 'function') {\n (eventStore as any).setRelays(this.relayUrls);\n }\n }\n\n /**\n * Set relay URLs\n */\n setRelays(urls: string[]): void {\n this.relayUrls = this.validateRelayUrls(urls);\n if (this.eventStore && 'setRelays' in this.eventStore && typeof this.eventStore.setRelays === 'function') {\n (this.eventStore as any).setRelays(this.relayUrls);\n }\n }\n\n /**\n * Get current relay URLs\n */\n getRelays(): string[] {\n return [...this.relayUrls];\n }\n\n /**\n * Publish an event to relays\n */\n async publishEvent(event: NostrEvent, timeoutMs = 1000): Promise<boolean> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('publish', this.minPublishIntervalMs);\n return new Promise((resolve, reject) => {\n if (this.relayUrls.length === 0) {\n reject(new Error('No relays configured'));\n return;\n }\n\n // Use EventStore's publish method\n // Note: This is a simplified implementation - actual applesauce API may differ\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.publish !== 'function') {\n reject(new Error('EventStore does not support publish method'));\n return;\n }\n const subscription = eventStore.publish(event).subscribe({\n next: (response: any) => {\n if (response?.type === 'OK') {\n subscription.unsubscribe();\n resolve(true);\n }\n },\n error: (error: Error) => {\n subscription.unsubscribe();\n reject(error);\n },\n });\n\n // Timeout fallback\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(false);\n }, timeoutMs);\n });\n }\n\n /**\n * Fetch a profile (Kind 0 event)\n */\n async fetchProfile(pubkey: string): Promise<ProfileMetadata | null> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-profile', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [0],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(null);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n subscription.unsubscribe();\n resolve(metadata);\n return;\n }\n console.error('Failed to parse profile metadata');\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(null);\n },\n error: (error: Error) => {\n console.error('Error fetching profile:', error);\n subscription.unsubscribe();\n resolve(null);\n },\n });\n\n // Timeout\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(null);\n }, 5000);\n });\n }\n\n /**\n * Fetch role tag from profile event (Kind 0)\n */\n async fetchProfileRoleTag(pubkey: string): Promise<string | null> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-role-tag', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [0],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(null);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0) {\n const tags = packet.event.tags || [];\n for (const tag of tags) {\n if (tag[0] === 'role' && tag[1]) {\n const candidate = tag[1].trim();\n if (isValidRoleTag(candidate)) {\n subscription.unsubscribe();\n resolve(candidate);\n return;\n }\n }\n }\n subscription.unsubscribe();\n resolve(null);\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(null);\n },\n error: (error: Error) => {\n console.error('Error fetching profile role tag:', error);\n subscription.unsubscribe();\n resolve(null);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(null);\n }, 5000);\n });\n }\n\n /**\n * Fetch a follow list (Kind 3 event)\n */\n async fetchFollowList(pubkey: string): Promise<FollowEntry[]> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-follow-list', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [3],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve([]);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 3) {\n const followList: FollowEntry[] = [];\n const tags = packet.event.tags || [];\n\n for (const tag of tags) {\n if (tag[0] === 'p' && tag[1]) {\n followList.push({\n pubkey: tag[1],\n relay: tag[2] || undefined,\n petname: tag[3] || undefined,\n });\n }\n }\n\n subscription.unsubscribe();\n resolve(followList);\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve([]);\n },\n error: (error: Error) => {\n console.error('Error fetching follow list:', error);\n subscription.unsubscribe();\n resolve([]);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve([]);\n }, 10000);\n });\n }\n\n /**\n * Fetch multiple profiles in batch\n */\n async fetchMultipleProfiles(pubkeys: string[]): Promise<Map<string, ProfileMetadata>> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n if (pubkeys.length === 0) {\n return new Map();\n }\n\n await this.enforceRateLimit('fetch-multiple-profiles', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const profiles = new Map<string, ProfileMetadata>();\n const filter = {\n kinds: [0],\n authors: pubkeys,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(new Map());\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0 && packet.event.pubkey) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n profiles.set(packet.event.pubkey, metadata);\n } else {\n console.error('Failed to parse profile metadata');\n }\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(profiles);\n },\n error: (error: Error) => {\n console.error('Error fetching profiles:', error);\n subscription.unsubscribe();\n resolve(profiles);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(profiles);\n }, 1000);\n });\n }\n\n /**\n * Query kind 0 events (profiles) by pubkey\n * If pubkeys array is empty, fetches recent kind 0 events\n */\n async queryProfiles(pubkeys: string[] = [], limit = 100): Promise<Map<string, ProfileMetadata>> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('query-profiles', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const profiles = new Map<string, { metadata: ProfileMetadata; timestamp: number }>();\n const filter: any = {\n kinds: [0],\n limit,\n };\n\n if (pubkeys.length > 0) {\n filter.authors = pubkeys;\n }\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(new Map());\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0 && packet.event.pubkey) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n const timestamp = packet.event.created_at || 0;\n const existing = profiles.get(packet.event.pubkey);\n if (!existing || timestamp > existing.timestamp) {\n profiles.set(packet.event.pubkey, { metadata, timestamp });\n }\n } else {\n console.error('Failed to parse profile metadata');\n }\n }\n },\n complete: () => {\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n },\n error: (error: Error) => {\n console.error('Error querying profiles:', error);\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n }, 10000);\n });\n }\n\n /**\n * Publish or update a kind 3 event (follow list/contacts)\n */\n async publishFollowList(\n pubkey: string,\n followList: FollowEntry[],\n signEvent: (event: NostrEvent) => Promise<NostrEvent>\n ): Promise<boolean> {\n const tags: string[][] = followList.map((entry) => {\n if (!isValidPubkey(entry.pubkey)) {\n throw new Error('Invalid pubkey format for follow list entry.');\n }\n if (entry.relay && !isValidRelayUrl(entry.relay)) {\n throw new Error('Invalid relay URL format for follow list entry.');\n }\n const tag: string[] = ['p', entry.pubkey];\n if (entry.relay) {\n tag.push(entry.relay);\n }\n if (entry.petname) {\n tag.push(entry.petname);\n }\n return tag;\n });\n\n const event: NostrEvent = {\n kind: 3,\n content: '',\n created_at: Math.floor(Date.now() / 1000),\n tags,\n };\n\n const signedEvent = await signEvent(event);\n return await this.publishEvent(signedEvent);\n }\n\n private validateRelayUrls(urls: string[]): string[] {\n if (urls.length === 0) {\n return [];\n }\n const validUrls = urls.filter((url) => isValidRelayUrl(url));\n if (validUrls.length !== urls.length) {\n throw new Error('Invalid relay URL format');\n }\n return validUrls;\n }\n\n private parseProfileMetadata(content: string): ProfileMetadata | null {\n if (content.length > this.maxProfileContentSize) {\n return null;\n }\n try {\n const parsed = JSON.parse(content);\n if (!parsed || typeof parsed !== 'object') {\n return null;\n }\n const metadata: ProfileMetadata = {};\n if (typeof (parsed as ProfileMetadata).name === 'string') {\n metadata.name = (parsed as ProfileMetadata).name;\n }\n if (typeof (parsed as ProfileMetadata).display_name === 'string') {\n metadata.display_name = (parsed as ProfileMetadata).display_name;\n }\n if (typeof (parsed as ProfileMetadata).about === 'string') {\n metadata.about = (parsed as ProfileMetadata).about;\n }\n if (typeof (parsed as ProfileMetadata).picture === 'string') {\n metadata.picture = (parsed as ProfileMetadata).picture;\n }\n if (typeof (parsed as ProfileMetadata).website === 'string') {\n metadata.website = (parsed as ProfileMetadata).website;\n }\n return metadata;\n } catch (error) {\n console.error('Failed to parse profile metadata:', error);\n return null;\n }\n }\n\n private async enforceRateLimit(action: string, minIntervalMs: number): Promise<void> {\n const lastAt = this.lastActionAt.get(action) ?? 0;\n const now = Date.now();\n const waitMs = minIntervalMs - (now - lastAt);\n if (waitMs > 0) {\n await new Promise((resolve) => setTimeout(resolve, waitMs));\n }\n this.lastActionAt.set(action, Date.now());\n }\n}\n\n","import { useState } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { AuthState } from '../../types/auth';\nimport './Auth.css';\n\ninterface LoginButtonProps {\n authService: AuthService;\n setAuthenticated: AuthState['setAuthenticated'];\n setLoginError: AuthState['setLoginError'];\n onSuccess?: () => void;\n}\n\nexport function LoginButton({\n authService,\n setAuthenticated,\n setLoginError,\n onSuccess,\n}: LoginButtonProps) {\n const [isLoading, setIsLoading] = useState(false);\n\n const handleLogin = async () => {\n setIsLoading(true);\n setLoginError(null);\n\n try {\n if (!authService.hasKeyInfo()) {\n throw new Error('No account found. Please register first.');\n }\n\n const keyInfo = authService.getCurrentKeyInfo();\n if (!keyInfo) {\n throw new Error('Failed to load account information.');\n }\n\n await authService.getPublicKey();\n setAuthenticated(keyInfo);\n\n if (onSuccess) {\n onSuccess();\n }\n } catch (err) {\n console.error('Login error:', err);\n const errorMessage = err instanceof Error ? err.message : 'Failed to login';\n setLoginError(errorMessage);\n setIsLoading(false);\n }\n };\n\n return (\n <div className=\"login-button-container\">\n <button\n className=\"auth-button secondary\"\n onClick={handleLogin}\n disabled={isLoading}\n >\n {isLoading ? 'Logging in...' : 'Login'}\n </button>\n </div>\n );\n}\n\n","import { useState } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { AuthState } from '../../types/auth';\nimport './Auth.css';\n\ninterface RegistrationFlowProps {\n authService: AuthService;\n setAuthenticated: AuthState['setAuthenticated'];\n onSuccess?: () => void;\n}\n\nexport function RegistrationFlow({\n authService,\n setAuthenticated,\n onSuccess,\n}: RegistrationFlowProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n const [step, setStep] = useState<'info' | 'creating' | 'success'>('info');\n const [username, setUsername] = useState('');\n const maxUsernameLength = 100;\n\n const handleRegister = async () => {\n setIsLoading(true);\n setError(null);\n setStep('creating');\n\n try {\n authService.clearStoredKeyInfo();\n\n let credentialId: Uint8Array;\n const passkeyUsername = username.trim() || undefined;\n \n try {\n credentialId = await authService.createPasskey(passkeyUsername);\n } catch (passkeyError) {\n console.error('[RegistrationFlow] Passkey creation failed:', passkeyError);\n throw new Error('Unable to create passkey. Please try again.');\n }\n\n let keyInfo;\n try {\n keyInfo = await authService.createNostrKey(credentialId);\n } catch (nostrKeyError) {\n console.error('[RegistrationFlow] Failed to create Nostr key from passkey:', nostrKeyError);\n throw new Error('Unable to finalize account. Please try again.');\n }\n\n authService.setCurrentKeyInfo(keyInfo);\n setAuthenticated(keyInfo);\n setStep('success');\n\n if (onSuccess) {\n setTimeout(() => {\n onSuccess();\n }, 1500);\n }\n } catch (err) {\n console.error('[RegistrationFlow] Registration error:', err);\n setError('Registration failed. Please try again.');\n setStep('info');\n setIsLoading(false);\n }\n };\n\n return (\n <div className=\"auth-container\">\n <div className=\"auth-card\">\n <h1>Create Account</h1>\n <p className=\"auth-description\">\n Create a new decentralized Identity. Your identity will be securely derived from a passkey.\n </p>\n\n {step === 'info' && (\n <>\n <div className=\"auth-features\">\n <div className=\"feature-item\">\n <span className=\"feature-icon\">🔐</span>\n <span>Phishing-resistant authentication</span>\n </div>\n <div className=\"feature-item\">\n <span className=\"feature-icon\">📱</span>\n <span>Biometric authentication support</span>\n </div>\n <div className=\"feature-item\">\n <span className=\"feature-icon\">🌐</span>\n <span>Cross-device synchronization</span>\n </div>\n </div>\n\n <div className=\"username-section\">\n <label htmlFor=\"username\" className=\"username-label\">\n Name (Optional)\n </label>\n <input\n id=\"username\"\n type=\"text\"\n className=\"username-input\"\n placeholder=\"Enter a name for this passkey\"\n value={username}\n onChange={(e) => setUsername(e.target.value)}\n disabled={isLoading}\n maxLength={maxUsernameLength}\n />\n </div>\n\n {error && <div className=\"error-message\">{error}</div>}\n\n <button\n className=\"auth-button primary\"\n onClick={handleRegister}\n disabled={isLoading}\n >\n {isLoading ? 'Creating...' : 'Create Account'}\n </button>\n </>\n )}\n\n {step === 'creating' && (\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Creating your passkey...</p>\n <p className=\"loading-hint\">Please follow your browser's authentication prompt</p>\n <p className=\"loading-hint-small\">\n 💡 Using your system's native passkey manager (Touch ID, Face ID, Windows Hello, etc.)\n </p>\n </div>\n )}\n\n {step === 'success' && (\n <div className=\"success-state\">\n <div className=\"success-icon\">✓</div>\n <p>Account created successfully!</p>\n <p className=\"success-hint\">Redirecting to profile setup...</p>\n </div>\n )}\n </div>\n </div>\n );\n}\n\n","import DOMPurify from 'isomorphic-dompurify';\n\nconst sanitizeOptions = {\n ALLOWED_TAGS: [],\n ALLOWED_ATTR: [],\n};\n\nexport const sanitizeText = (value: string): string =>\n DOMPurify.sanitize(value, sanitizeOptions);\n","import { useState } from 'react';\nimport { useZxing } from 'react-zxing';\n\ninterface BarcodeScannerProps {\n /** Called when a code is successfully read */\n onDecode: (value: string) => void;\n /** Optional flag to hide the scanner */\n active?: boolean;\n}\n\nexport const BarcodeScanner = ({ onDecode, active = true }: BarcodeScannerProps) => {\n const [error, setError] = useState<string | null>(null);\n const { ref } = useZxing({\n onDecodeResult: (result) => {\n onDecode(result.getText());\n },\n onError: (e: unknown) => {\n const errorMessage = e instanceof Error ? e.message : 'Camera error';\n setError(errorMessage);\n },\n });\n\n if (!active) return null;\n\n return (\n <div style={{ position: 'relative', width: '100%', maxWidth: '400px' }}>\n <video\n ref={ref as React.LegacyRef<HTMLVideoElement>}\n style={{ width: '100%', borderRadius: '8px' }}\n playsInline\n muted\n />\n {error && (\n <p style={{ color: 'red', marginTop: '0.5rem' }}>\n {error}\n </p>\n )}\n </div>\n );\n};\n\n","import { useState, useEffect } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { RelayService } from '../../services/relay.service';\nimport type { AuthState } from '../../types/auth';\nimport type { ProfileMetadata, FollowEntry } from '../../types/nostr';\nimport { sanitizeText } from '../../utils/sanitize';\nimport { isValidHttpUrl } from '../../utils/validation';\nimport { BarcodeScanner } from './BarcodeScanner';\nimport './Membership.css';\n\ninterface ProfileWithPubkey extends ProfileMetadata {\n pubkey: string;\n}\n\ninterface MembershipPageProps {\n authService: AuthService;\n relayService: RelayService;\n publicKey: string | null;\n onUnauthenticated?: () => void;\n}\n\nconst TrustInfo = () => (\n <section className=\"trust-info\">\n <h2>Building Trust with Verifiable Relationship Credentials</h2>\n <p>\n Beyond the Personhood Credential, members can create <strong>Verifiable Relationship\n Credentials (VRCs)</strong>. A VRC is issued directly between two members – for example,\n by scanning a QR‑code at a meetup – and certifies a first‑hand trust link.\n </p>\n <p>\n Each VRC becomes a node in a decentralized trust graph. When you add a member, the\n system records a <strong>role</strong> tag that ties the new\n participant to your existing graph, enabling permissionless access to network‑state\n resources while keeping the underlying data private.\n </p>\n <p>\n The graph grows organically: trusted authorities issue PHCs, and members continuously\n enrich the network with peer‑generated VRCs, creating a resilient, scalable web of\n verified participants.\n </p>\n </section>\n);\n\nexport function MembershipPage({\n authService,\n relayService,\n publicKey,\n onUnauthenticated,\n}: MembershipPageProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [isSaving, setIsSaving] = useState(false);\n const [saveMessage, setSaveMessage] = useState<string | null>(null);\n const [searchQuery, setSearchQuery] = useState('');\n const [profiles, setProfiles] = useState<ProfileWithPubkey[]>([]);\n const [members, setMembers] = useState<FollowEntry[]>([]);\n const [memberProfiles, setMemberProfiles] = useState<Map<string, ProfileMetadata>>(new Map());\n const [showScanner, setShowScanner] = useState(false);\n\n useEffect(() => {\n if (!publicKey) {\n if (onUnauthenticated) {\n onUnauthenticated();\n }\n return;\n }\n\n loadFollowList();\n }, [publicKey]);\n\n useEffect(() => {\n if (members.length > 0) {\n loadMemberProfiles();\n }\n }, [members]);\n\n const handleDecoded = (decoded: string) => {\n setSearchQuery(decoded);\n setShowScanner(false);\n handleSearch();\n };\n\n const loadFollowList = async () => {\n if (!publicKey) return;\n\n setIsLoading(true);\n try {\n const followList = await relayService.fetchFollowList(publicKey);\n setMembers(followList);\n } catch (error) {\n console.error('Failed to load follow list:', error);\n setSaveMessage('Failed to load membership list');\n } finally {\n setIsLoading(false);\n }\n };\n\n const loadMemberProfiles = async () => {\n if (members.length === 0) return;\n\n try {\n const pubkeys = members.map((m) => m.pubkey);\n const profilesMap = await relayService.fetchMultipleProfiles(pubkeys);\n setMemberProfiles(profilesMap);\n } catch (error) {\n console.error('Failed to load member profiles:', error);\n }\n };\n\n const handleSearch = async () => {\n if (!searchQuery.trim()) {\n setIsLoading(true);\n try {\n const profilesMap = await relayService.queryProfiles([], 50);\n const profilesList: ProfileWithPubkey[] = [];\n profilesMap.forEach((profile, pubkey) => {\n profilesList.push({ ...profile, pubkey });\n });\n setProfiles(profilesList);\n } catch (error) {\n console.error('Failed to query profiles:', error);\n setSaveMessage('Failed to search profiles');\n } finally {\n setIsLoading(false);\n }\n return;\n }\n\n const trimmedQuery = searchQuery.trim();\n if (trimmedQuery.length !== 64 || !/^[0-9a-fA-F]+$/.test(trimmedQuery)) {\n setSaveMessage('Invalid pubkey format. Must be 64 hex characters.');\n setTimeout(() => setSaveMessage(null), 3000);\n return;\n }\n\n setIsLoading(true);\n try {\n const profilesMap = await relayService.queryProfiles([trimmedQuery], 1);\n const profilesList: ProfileWithPubkey[] = [];\n profilesMap.forEach((profile, pubkey) => {\n profilesList.push({ ...profile, pubkey });\n });\n setProfiles(profilesList);\n if (profilesList.length === 0) {\n setSaveMessage('No profile found for this pubkey');\n setTimeout(() => setSaveMessage(null), 3000);\n }\n } catch (error) {\n console.error('Failed to query profile:', error);\n setSaveMessage('Failed to search profile');\n } finally {\n setIsLoading(false);\n }\n };\n\n const handleAddMember = async (pubkey: string) => {\n if (!publicKey) return;\n\n if (members.some((m) => m.pubkey === pubkey)) {\n setSaveMessage('User is already a member');\n setTimeout(() => setSaveMessage(null), 3000);\n return;\n }\n\n setIsSaving(true);\n try {\n const newMembers: FollowEntry[] = [\n ...members,\n {\n pubkey,\n },\n ];\n\n await relayService.publishFollowList(publicKey, newMembers, (event) =>\n authService.signEvent(event)\n );\n\n setMembers(newMembers);\n setSaveMessage('Member added successfully!');\n setTimeout(() => setSaveMessage(null), 3000);\n } catch (error) {\n console.error('Failed to add member:', error);\n setSaveMessage('Failed to add member. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const handleRemoveMember = async (pubkey: string) => {\n if (!publicKey) return;\n\n setIsSaving(true);\n try {\n const newMembers = members.filter((m) => m.pubkey !== pubkey);\n\n await relayService.publishFollowList(publicKey, newMembers, (event) =>\n authService.signEvent(event)\n );\n\n setMembers(newMembers);\n setSaveMessage('Member removed successfully!');\n setTimeout(() => setSaveMessage(null), 3000);\n } catch (error) {\n console.error('Failed to remove member:', error);\n setSaveMessage('Failed to remove member. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const getProfileDisplayName = (profile: ProfileMetadata, pubkey: string): string => {\n return profile.display_name || profile.name || pubkey.slice(0, 16) + '...';\n };\n\n const formatPubkey = (pubkey: string): string => {\n return `${pubkey.slice(0, 8)}...${pubkey.slice(-8)}`;\n };\n\n if (isLoading && members.length === 0) {\n return (\n <div className=\"membership-container\">\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Loading membership list...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"membership-container\">\n <div className=\"membership-card\">\n <h1>Membership Management</h1>\n <TrustInfo />\n <p className=\"membership-description\">\n Query applicants and manage the membership list.\n </p>\n\n {saveMessage && (\n <div\n className={`save-message ${\n saveMessage.includes('Error') || saveMessage.includes('Failed')\n ? 'error'\n : 'success'\n }`}\n >\n {saveMessage}\n </div>\n )}\n\n <div className=\"search-section\">\n <h2>Add Member</h2>\n <div className=\"search-form\">\n <input\n type=\"text\"\n value={searchQuery}\n onChange={(e) => setSearchQuery(e.target.value)}\n onKeyPress={(e) => e.key === 'Enter' && handleSearch()}\n placeholder=\"Enter pubkey or leave empty for recent profiles\"\n className=\"search-input\"\n disabled={isLoading}\n />\n <button\n onClick={handleSearch}\n className=\"search-button\"\n disabled={isLoading || isSaving}\n >\n {isLoading ? 'Searching...' : 'Search'}\n </button>\n <button\n onClick={() => setShowScanner((prev) => !prev)}\n className=\"scanner-toggle\"\n disabled={isLoading || isSaving}\n >\n {showScanner ? 'Close Scanner' : 'Scan QR'}\n </button>\n </div>\n\n {showScanner && (\n <div style={{ marginTop: '1rem' }}>\n <BarcodeScanner onDecode={handleDecoded} active={true} />\n <p style={{ fontSize: '0.85rem', marginTop: '0.5rem' }}>\n Point at QR‑code containing a pubkey\n </p>\n </div>\n )}\n\n {profiles.length > 0 && (\n <div className=\"profiles-list\">\n <h3>Search Results</h3>\n {profiles.map((profile) => {\n const isMember = members.some((m) => m.pubkey === profile.pubkey);\n return (\n <div key={profile.pubkey} className=\"profile-item\">\n <div className=\"profile-info\">\n {profile.picture && isValidHttpUrl(profile.picture) && (\n <img\n src={profile.picture}\n alt={sanitizeText(getProfileDisplayName(profile, profile.pubkey))}\n className=\"profile-avatar\"\n loading=\"lazy\"\n referrerPolicy=\"no-referrer\"\n onError={(event) => {\n event.currentTarget.style.display = 'none';\n }}\n />\n )}\n <div className=\"profile-details\">\n <div className=\"profile-name\">\n {sanitizeText(getProfileDisplayName(profile, profile.pubkey))}\n </div>\n <div className=\"profile-pubkey\">{formatPubkey(profile.pubkey)}</div>\n {profile.about && (\n <div className=\"profile-about\">{sanitizeText(profile.about)}</div>\n )}\n </div>\n </div>\n <button\n onClick={() =>\n isMember\n ? handleRemoveMember(profile.pubkey)\n : handleAddMember(profile.pubkey)\n }\n className={`member-button ${isMember ? 'remove' : 'add'}`}\n disabled={isSaving}\n >\n {isMember ? 'Remove' : 'Add Member'}\n </button>\n </div>\n );\n })}\n </div>\n )}\n </div>\n\n <div className=\"members-section\">\n <h2>Current Members ({members.length})</h2>\n {members.length === 0 ? (\n <p className=\"empty-message\">No members yet. Add members from search results above.</p>\n ) : (\n <div className=\"members-list\">\n {members.map((member) => {\n const profile = memberProfiles.get(member.pubkey);\n return (\n <div key={member.pubkey} className=\"member-item\">\n <div className=\"profile-info\">\n {profile?.picture && isValidHttpUrl(profile.picture) && (\n <img\n src={profile.picture}\n alt={sanitizeText(getProfileDisplayName(profile || {}, member.pubkey))}\n className=\"profile-avatar\"\n loading=\"lazy\"\n referrerPolicy=\"no-referrer\"\n onError={(event) => {\n event.currentTarget.style.display = 'none';\n }}\n />\n )}\n <div className=\"profile-details\">\n <div className=\"profile-name\">\n {profile\n ? sanitizeText(getProfileDisplayName(profile, member.pubkey))\n : formatPubkey(member.pubkey)}\n </div>\n <div className=\"profile-pubkey\">{formatPubkey(member.pubkey)}</div>\n {profile?.about && (\n <div className=\"profile-about\">{sanitizeText(profile.about)}</div>\n )}\n {member.petname && (\n <div className=\"profile-petname\">Name: {sanitizeText(member.petname)}</div>\n )}\n </div>\n </div>\n <button\n onClick={() => handleRemoveMember(member.pubkey)}\n className=\"member-button remove\"\n disabled={isSaving}\n >\n Remove\n </button>\n </div>\n );\n })}\n </div>\n )}\n </div>\n </div>\n </div>\n );\n}\n\n","import { useState, useEffect } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { RelayService } from '../../services/relay.service';\nimport type { AuthState } from '../../types/auth';\nimport type { ProfileMetadata } from '../../types/nostr';\nimport { sanitizeText } from '../../utils/sanitize';\nimport { isValidPubkey, isValidRoleTag } from '../../utils/validation';\nimport './Profile.css';\n\ninterface ProfilePageProps {\n authService: AuthService;\n relayService: RelayService;\n publicKey: string | null;\n onUnauthenticated?: () => void;\n onSuccess?: () => void;\n onRoleSuggestion?: (about: string) => Promise<string | null>;\n}\n\nconst PersonhoodInfo = () => (\n <section className=\"personhood-info\">\n <p>\n Network‑state members receive a <strong>Personhood Credential (PHC)</strong> from a trusted authority.\n The PHC attests that the holder is a unique, real individual. Because the credential lives\n in your passport, you can present a <em>zero‑knowledge proof</em> that you are verified.\n </p>\n <p>\n In this form we compare the name you enter with the name disclosed by your verified\n passport proof. If the two match, the profile will be saved as your business card credential together with a\n passport tag that references the PHC's unique identifier.\n </p>\n </section>\n);\n\nconst MAX_NAME_LENGTH = 100;\nconst MAX_ABOUT_LENGTH = 1000;\nconst MAX_URL_LENGTH = 2048;\n\nexport function ProfilePage({\n authService,\n relayService,\n publicKey,\n onUnauthenticated,\n onSuccess,\n onRoleSuggestion,\n}: ProfilePageProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [isSaving, setIsSaving] = useState(false);\n const [saveMessage, setSaveMessage] = useState<string | null>(null);\n const [suggestedRole, setSuggestedRole] = useState<string | null>(null);\n const [isGettingRole, setIsGettingRole] = useState(false);\n const [formData, setFormData] = useState<ProfileMetadata>({\n name: '',\n display_name: '',\n about: '',\n picture: '',\n website: '',\n });\n\n useEffect(() => {\n if (!publicKey) {\n if (onUnauthenticated) {\n onUnauthenticated();\n }\n return;\n }\n\n loadProfile();\n }, [publicKey]);\n\n const loadProfile = async () => {\n if (!publicKey) return;\n\n setIsLoading(true);\n try {\n const [profile, roleTag] = await Promise.all([\n relayService.fetchProfile(publicKey),\n relayService.fetchProfileRoleTag(publicKey),\n ]);\n \n if (profile) {\n setFormData({\n name: sanitizeText(profile.name || ''),\n display_name: sanitizeText(profile.display_name || ''),\n about: sanitizeText(profile.about || ''),\n picture: profile.picture || '',\n website: profile.website || '',\n });\n }\n \n if (roleTag) {\n setSuggestedRole(roleTag);\n }\n } catch (error) {\n console.error('Failed to load profile:', error);\n } finally {\n setIsLoading(false);\n }\n };\n\n const handleSubmit = async (e: React.FormEvent) => {\n e.preventDefault();\n if (!publicKey) return;\n\n setIsSaving(true);\n setSaveMessage(null);\n\n try {\n const tags: string[][] = [];\n if (!isValidPubkey(publicKey)) {\n throw new Error('Invalid public key format');\n }\n\n // Suggest role based on \"about\" field if it has content and callback provided\n if (formData.about && formData.about.trim().length > 0 && onRoleSuggestion) {\n setIsGettingRole(true);\n try {\n const roleSuggestion = await onRoleSuggestion(formData.about);\n const candidate = roleSuggestion ? roleSuggestion.trim() : '';\n if (candidate && isValidRoleTag(candidate)) {\n tags.push(['role', candidate]);\n setSuggestedRole(candidate);\n } else {\n setSuggestedRole(null);\n }\n } catch (error) {\n console.error('Failed to get role suggestion:', error);\n setSuggestedRole(null);\n } finally {\n setIsGettingRole(false);\n }\n } else {\n setSuggestedRole(null);\n }\n\n const profileEvent = {\n kind: 0,\n content: JSON.stringify({\n ...formData,\n name: sanitizeText(formData.name || ''),\n display_name: sanitizeText(formData.display_name || ''),\n about: sanitizeText(formData.about || ''),\n }),\n created_at: Math.floor(Date.now() / 1000),\n tags,\n };\n\n const follows = {\n kind: 3,\n content: \"\",\n created_at: Math.floor(Date.now() / 1000),\n tags: [\n (() => {\n const followTag: string[] = ['p', publicKey];\n const relayUrl = relayService.getRelays()[0];\n if (relayUrl) {\n followTag.push(relayUrl);\n }\n if (formData.name) {\n followTag.push(formData.name);\n }\n return followTag;\n })(),\n ],\n };\n\n const signedProfile = await authService.signEvent(profileEvent);\n const signedFollows = await authService.signEvent(follows);\n\n await relayService.publishEvent(signedProfile);\n await relayService.publishEvent(signedFollows);\n\n setSaveMessage('Profile saved successfully!');\n setTimeout(() => {\n setSaveMessage(null);\n }, 3000);\n\n if (onSuccess) {\n onSuccess();\n }\n } catch (error) {\n console.error('Failed to save profile:', error);\n setSaveMessage('Failed to save profile. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const handleChange = (\n e: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>\n ) => {\n const { name, value } = e.target;\n setFormData((prev) => ({\n ...prev,\n [name]: value,\n }));\n };\n\n if (isLoading) {\n return (\n <div className=\"profile-container\">\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Loading profile...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"profile-container\">\n <div className=\"profile-card\">\n <h1>Profile Setup</h1>\n <PersonhoodInfo />\n\n <form onSubmit={handleSubmit} className=\"profile-form\">\n <div className=\"form-group\">\n <label htmlFor=\"name\">First Name</label>\n <input\n type=\"text\"\n id=\"name\"\n name=\"name\"\n value={formData.name || ''}\n onChange={handleChange}\n placeholder=\"Your username\"\n maxLength={MAX_NAME_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"display_name\">Last Name</label>\n <input\n type=\"text\"\n id=\"display_name\"\n name=\"display_name\"\n value={formData.display_name || ''}\n onChange={handleChange}\n placeholder=\"Your Last Name\"\n maxLength={MAX_NAME_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"about\">About</label>\n <textarea\n id=\"about\"\n name=\"about\"\n value={formData.about || ''}\n onChange={handleChange}\n placeholder=\"Tell us about yourself\"\n rows={4}\n maxLength={MAX_ABOUT_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"picture\">Profile Picture URL</label>\n <input\n type=\"url\"\n id=\"picture\"\n name=\"picture\"\n value={formData.picture || ''}\n onChange={handleChange}\n placeholder=\"https://example.com/avatar.jpg\"\n maxLength={MAX_URL_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"website\">LinkedIn</label>\n <input\n type=\"url\"\n id=\"website\"\n name=\"website\"\n value={formData.website || ''}\n onChange={handleChange}\n placeholder=\"https://linkedin.com/example\"\n maxLength={MAX_URL_LENGTH}\n />\n </div>\n\n {saveMessage && (\n <div className={`save-message ${saveMessage.includes('Error') ? 'error' : 'success'}`}>\n {saveMessage}\n </div>\n )}\n\n {(isGettingRole || suggestedRole) && (\n <div className=\"role-tag-container\">\n {isGettingRole ? (\n <>\n <div className=\"role-tag-label\">Getting AI suggestion...</div>\n <div className=\"role-tag-loading\">\n <div className=\"role-tag-spinner\"></div>\n </div>\n </>\n ) : suggestedRole ? (\n <>\n <div className=\"role-tag-label\">AI Suggested Role:</div>\n <div className=\"role-tag\">\n {sanitizeText(suggestedRole)}\n </div>\n </>\n ) : null}\n </div>\n )}\n\n <div className=\"form-actions\">\n <button type=\"submit\" className=\"save-button\" disabled={isSaving}>\n {isSaving ? 'Saving...' : 'Save Profile'}\n </button>\n </div>\n </form>\n </div>\n </div>\n );\n}\n\n","import { useEffect } from 'react';\nimport type { AuthService } from '../services/auth.service';\nimport type { AuthState } from '../types/auth';\n\n/**\n * Hook to initialize auth state on app load\n */\nexport function useAuthInit(\n authService: AuthService,\n setAuthenticated: AuthState['setAuthenticated']\n): void {\n useEffect(() => {\n if (authService.hasKeyInfo()) {\n const keyInfo = authService.getCurrentKeyInfo();\n if (keyInfo) {\n setAuthenticated(keyInfo);\n }\n }\n }, [authService, setAuthenticated]);\n}\n\n","import { create } from 'zustand';\nimport type { NostrKeyInfo } from '../utils';\nimport type { AuthState } from '../types/auth';\n\nexport const createAuthStore = () => {\n return create<AuthState>((set) => ({\n isAuthenticated: false,\n publicKey: null,\n keyInfo: null,\n loginError: null,\n setAuthenticated: (keyInfo: NostrKeyInfo | null) => {\n set({\n isAuthenticated: !!keyInfo,\n publicKey: keyInfo?.pubkey || null,\n keyInfo,\n loginError: null,\n });\n },\n setLoginError: (error: string | null) => {\n set({ loginError: error });\n },\n logout: () => {\n set({\n isAuthenticated: false,\n publicKey: null,\n keyInfo: null,\n loginError: null,\n });\n },\n }));\n};\n\n// Default singleton instance\nexport const useAuthStore = createAuthStore();\n\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","names":["NosskeyManager","DOMPurify"],"sources":["../src/services/auth.service.ts","../src/utils/validation.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/utils/sanitize.ts","../src/components/membership/BarcodeScanner.tsx","../src/components/membership/MembershipPage.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":["import { NosskeyManager, type NostrKeyInfo, type NostrEvent, type PasskeyCreationOptions } from '../utils';\nimport type { AuthServiceConfig } from '../types/auth';\n\n/**\n * Service wrapper around NosskeyManager\n * Handles WebAuthn/Passkey integration with Nostr\n */\nexport class AuthService {\n private manager: NosskeyManager | null = null;\n private config: AuthServiceConfig;\n\n constructor(config: AuthServiceConfig = {}) {\n this.config = {\n rpId: config.rpId || (typeof window !== 'undefined' ? window.location.hostname.replace(/^www\\./, '') : 'localhost'),\n rpName: config.rpName || this.getDefaultRpName(),\n storageKey: config.storageKey || 'nsauth_keyinfo',\n cacheTimeoutMs: config.cacheTimeoutMs || 30 * 60 * 1000,\n cacheOnCreation: config.cacheOnCreation !== undefined ? config.cacheOnCreation : true,\n };\n }\n\n private getDefaultRpName(): string {\n if (typeof window === 'undefined') return 'localhost';\n const hostname = window.location.hostname;\n if (hostname.includes('nosskey.app')) return 'nosskey.app';\n return hostname.replace(/^www\\./, '');\n }\n\n /**\n * Initialize the NosskeyManager instance\n */\n private getManager(): NosskeyManager {\n if (!this.manager) {\n this.manager = new NosskeyManager({\n cacheOptions: {\n enabled: true,\n timeoutMs: this.config.cacheTimeoutMs,\n cacheOnCreation: this.config.cacheOnCreation,\n },\n storageOptions: {\n enabled: true,\n storageKey: this.config.storageKey,\n },\n });\n }\n return this.manager;\n }\n\n /**\n * Create a new passkey\n * Uses platform authenticator only (Touch ID, Face ID, Windows Hello)\n */\n async createPasskey(username?: string): Promise<Uint8Array> {\n const manager = this.getManager();\n \n const rpId = this.config.rpId === 'localhost' ? 'localhost' : this.config.rpId;\n const rpName = this.config.rpName;\n \n const trimmedUsername = username?.trim();\n const uniqueUsername = trimmedUsername \n ? trimmedUsername \n : `user-${Date.now()}@example.com`;\n \n const options: PasskeyCreationOptions = {\n rp: {\n id: rpId,\n name: rpName,\n },\n user: {\n name: uniqueUsername,\n displayName: trimmedUsername || 'User',\n },\n authenticatorSelection: {\n authenticatorAttachment: 'platform',\n residentKey: 'preferred',\n userVerification: 'preferred',\n },\n extensions: {\n prf: {},\n },\n };\n \n return await manager.createPasskey(options);\n }\n\n /**\n * Create a new Nostr key from a credential ID\n */\n async createNostrKey(credentialId?: Uint8Array): Promise<NostrKeyInfo> {\n const manager = this.getManager();\n return await manager.createNostrKey(credentialId);\n }\n\n /**\n * Get the current public key\n */\n async getPublicKey(): Promise<string> {\n const manager = this.getManager();\n return await manager.getPublicKey();\n }\n\n /**\n * Sign a Nostr event\n */\n async signEvent(event: NostrEvent): Promise<NostrEvent> {\n const manager = this.getManager();\n return await manager.signEvent(event);\n }\n\n /**\n * Get current key info\n */\n getCurrentKeyInfo(): NostrKeyInfo | null {\n const manager = this.getManager();\n return manager.getCurrentKeyInfo();\n }\n\n /**\n * Set current key info\n */\n setCurrentKeyInfo(keyInfo: NostrKeyInfo): void {\n const manager = this.getManager();\n manager.setCurrentKeyInfo(keyInfo);\n }\n\n /**\n * Check if key info exists\n */\n hasKeyInfo(): boolean {\n const manager = this.getManager();\n return manager.hasKeyInfo();\n }\n\n /**\n * Clear stored key info\n */\n clearStoredKeyInfo(): void {\n const manager = this.getManager();\n manager.clearStoredKeyInfo();\n }\n\n /**\n * Check if PRF is supported\n */\n async isPrfSupported(): Promise<boolean> {\n const { isPrfSupported } = await import('../utils');\n return await isPrfSupported();\n }\n}\n\n","export const MAX_ROLE_LENGTH = 100;\nconst ROLE_PATTERN = /^[a-zA-Z0-9\\s\\-_]+$/;\n\nexport const isValidRoleTag = (role: string): boolean => {\n const trimmed = role.trim();\n return (\n trimmed.length > 0 &&\n trimmed.length <= MAX_ROLE_LENGTH &&\n ROLE_PATTERN.test(trimmed)\n );\n};\n\nexport const isValidPubkey = (pubkey: string): boolean =>\n pubkey.length === 64 && /^[0-9a-fA-F]+$/.test(pubkey);\n\nexport const isValidHttpUrl = (url: string): boolean => {\n try {\n const parsed = new URL(url);\n return ['http:', 'https:'].includes(parsed.protocol);\n } catch {\n return false;\n }\n};\n\nexport const isValidRelayUrl = (url: string): boolean => {\n try {\n const parsed = new URL(url);\n return ['ws:', 'wss:'].includes(parsed.protocol) && parsed.hostname.length > 0;\n } catch {\n return false;\n }\n};\n","import type { EventStore } from 'applesauce-core';\nimport type { NostrEvent } from '../types/nostr';\nimport type { ProfileMetadata, FollowEntry } from '../types/nostr';\nimport type { RelayServiceConfig } from '../types/auth';\nimport { isValidPubkey, isValidRelayUrl, isValidRoleTag } from '../utils/validation';\n\n/**\n * Service for communicating with Nostr relays using applesauce-core\n */\nexport class RelayService {\n private eventStore: EventStore | null = null;\n private relayUrls: string[];\n private defaultRelays = ['wss://relay.damus.io'];\n private readonly maxProfileContentSize = 10000;\n private readonly minProfileQueryIntervalMs = 300;\n private readonly minPublishIntervalMs = 750;\n private readonly lastActionAt = new Map<string, number>();\n\n constructor(config: RelayServiceConfig = {}) {\n this.relayUrls = this.validateRelayUrls(config.relayUrls ?? this.defaultRelays);\n }\n\n /**\n * Initialize with applesauce EventStore\n */\n initialize(eventStore: EventStore): void {\n this.eventStore = eventStore;\n // Set default relays if EventStore has a method for it\n if (eventStore && 'setRelays' in eventStore && typeof eventStore.setRelays === 'function') {\n (eventStore as any).setRelays(this.relayUrls);\n }\n }\n\n /**\n * Set relay URLs\n */\n setRelays(urls: string[]): void {\n this.relayUrls = this.validateRelayUrls(urls);\n if (this.eventStore && 'setRelays' in this.eventStore && typeof this.eventStore.setRelays === 'function') {\n (this.eventStore as any).setRelays(this.relayUrls);\n }\n }\n\n /**\n * Get current relay URLs\n */\n getRelays(): string[] {\n return [...this.relayUrls];\n }\n\n /**\n * Publish an event to relays\n */\n async publishEvent(event: NostrEvent, timeoutMs = 1000): Promise<boolean> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('publish', this.minPublishIntervalMs);\n return new Promise((resolve, reject) => {\n if (this.relayUrls.length === 0) {\n reject(new Error('No relays configured'));\n return;\n }\n\n // Use EventStore's publish method\n // Note: This is a simplified implementation - actual applesauce API may differ\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.publish !== 'function') {\n reject(new Error('EventStore does not support publish method'));\n return;\n }\n const subscription = eventStore.publish(event).subscribe({\n next: (response: any) => {\n if (response?.type === 'OK') {\n subscription.unsubscribe();\n resolve(true);\n }\n },\n error: (error: Error) => {\n subscription.unsubscribe();\n reject(error);\n },\n });\n\n // Timeout fallback\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(false);\n }, timeoutMs);\n });\n }\n\n /**\n * Fetch a profile (Kind 0 event)\n */\n async fetchProfile(pubkey: string): Promise<ProfileMetadata | null> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-profile', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [0],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(null);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n subscription.unsubscribe();\n resolve(metadata);\n return;\n }\n console.error('Failed to parse profile metadata');\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(null);\n },\n error: (error: Error) => {\n console.error('Error fetching profile:', error);\n subscription.unsubscribe();\n resolve(null);\n },\n });\n\n // Timeout\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(null);\n }, 5000);\n });\n }\n\n /**\n * Fetch role tag from profile event (Kind 0)\n */\n async fetchProfileRoleTag(pubkey: string): Promise<string | null> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-role-tag', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [0],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(null);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0) {\n const tags = packet.event.tags || [];\n for (const tag of tags) {\n if (tag[0] === 'role' && tag[1]) {\n const candidate = tag[1].trim();\n if (isValidRoleTag(candidate)) {\n subscription.unsubscribe();\n resolve(candidate);\n return;\n }\n }\n }\n subscription.unsubscribe();\n resolve(null);\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(null);\n },\n error: (error: Error) => {\n console.error('Error fetching profile role tag:', error);\n subscription.unsubscribe();\n resolve(null);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(null);\n }, 5000);\n });\n }\n\n /**\n * Fetch a follow list (Kind 3 event)\n */\n async fetchFollowList(pubkey: string): Promise<FollowEntry[]> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('fetch-follow-list', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const filter = {\n kinds: [3],\n authors: [pubkey],\n limit: 1,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve([]);\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 3) {\n const followList: FollowEntry[] = [];\n const tags = packet.event.tags || [];\n\n for (const tag of tags) {\n if (tag[0] === 'p' && tag[1]) {\n followList.push({\n pubkey: tag[1],\n relay: tag[2] || undefined,\n petname: tag[3] || undefined,\n });\n }\n }\n\n subscription.unsubscribe();\n resolve(followList);\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve([]);\n },\n error: (error: Error) => {\n console.error('Error fetching follow list:', error);\n subscription.unsubscribe();\n resolve([]);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve([]);\n }, 10000);\n });\n }\n\n /**\n * Fetch multiple profiles in batch\n */\n async fetchMultipleProfiles(pubkeys: string[]): Promise<Map<string, ProfileMetadata>> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n if (pubkeys.length === 0) {\n return new Map();\n }\n\n await this.enforceRateLimit('fetch-multiple-profiles', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const profiles = new Map<string, ProfileMetadata>();\n const filter = {\n kinds: [0],\n authors: pubkeys,\n };\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(new Map());\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0 && packet.event.pubkey) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n profiles.set(packet.event.pubkey, metadata);\n } else {\n console.error('Failed to parse profile metadata');\n }\n }\n },\n complete: () => {\n subscription.unsubscribe();\n resolve(profiles);\n },\n error: (error: Error) => {\n console.error('Error fetching profiles:', error);\n subscription.unsubscribe();\n resolve(profiles);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n resolve(profiles);\n }, 1000);\n });\n }\n\n /**\n * Query kind 0 events (profiles) by pubkey\n * If pubkeys array is empty, fetches recent kind 0 events\n */\n async queryProfiles(pubkeys: string[] = [], limit = 100): Promise<Map<string, ProfileMetadata>> {\n if (!this.eventStore) {\n throw new Error('RelayService not initialized. Call initialize() with an EventStore instance.');\n }\n\n await this.enforceRateLimit('query-profiles', this.minProfileQueryIntervalMs);\n return new Promise((resolve) => {\n const profiles = new Map<string, { metadata: ProfileMetadata; timestamp: number }>();\n const filter: any = {\n kinds: [0],\n limit,\n };\n\n if (pubkeys.length > 0) {\n filter.authors = pubkeys;\n }\n\n const eventStore = this.eventStore as any;\n if (!eventStore || typeof eventStore.query !== 'function') {\n resolve(new Map());\n return;\n }\n const subscription = eventStore.query(filter).subscribe({\n next: (packet: any) => {\n if (packet?.event && packet.event.kind === 0 && packet.event.pubkey) {\n const metadata = this.parseProfileMetadata(packet.event.content);\n if (metadata) {\n const timestamp = packet.event.created_at || 0;\n const existing = profiles.get(packet.event.pubkey);\n if (!existing || timestamp > existing.timestamp) {\n profiles.set(packet.event.pubkey, { metadata, timestamp });\n }\n } else {\n console.error('Failed to parse profile metadata');\n }\n }\n },\n complete: () => {\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n },\n error: (error: Error) => {\n console.error('Error querying profiles:', error);\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n },\n });\n\n setTimeout(() => {\n subscription.unsubscribe();\n const result = new Map<string, ProfileMetadata>();\n profiles.forEach((value, pubkey) => {\n result.set(pubkey, value.metadata);\n });\n resolve(result);\n }, 10000);\n });\n }\n\n /**\n * Publish or update a kind 3 event (follow list/contacts)\n */\n async publishFollowList(\n pubkey: string,\n followList: FollowEntry[],\n signEvent: (event: NostrEvent) => Promise<NostrEvent>\n ): Promise<boolean> {\n const tags: string[][] = followList.map((entry) => {\n if (!isValidPubkey(entry.pubkey)) {\n throw new Error('Invalid pubkey format for follow list entry.');\n }\n if (entry.relay && !isValidRelayUrl(entry.relay)) {\n throw new Error('Invalid relay URL format for follow list entry.');\n }\n const tag: string[] = ['p', entry.pubkey];\n if (entry.relay) {\n tag.push(entry.relay);\n }\n if (entry.petname) {\n tag.push(entry.petname);\n }\n return tag;\n });\n\n const event: NostrEvent = {\n kind: 3,\n content: '',\n created_at: Math.floor(Date.now() / 1000),\n tags,\n };\n\n const signedEvent = await signEvent(event);\n return await this.publishEvent(signedEvent);\n }\n\n private validateRelayUrls(urls: string[]): string[] {\n if (urls.length === 0) {\n return [];\n }\n const validUrls = urls.filter((url) => isValidRelayUrl(url));\n if (validUrls.length !== urls.length) {\n throw new Error('Invalid relay URL format');\n }\n return validUrls;\n }\n\n private parseProfileMetadata(content: string): ProfileMetadata | null {\n if (content.length > this.maxProfileContentSize) {\n return null;\n }\n try {\n const parsed = JSON.parse(content);\n if (!parsed || typeof parsed !== 'object') {\n return null;\n }\n const metadata: ProfileMetadata = {};\n if (typeof (parsed as ProfileMetadata).name === 'string') {\n metadata.name = (parsed as ProfileMetadata).name;\n }\n if (typeof (parsed as ProfileMetadata).display_name === 'string') {\n metadata.display_name = (parsed as ProfileMetadata).display_name;\n }\n if (typeof (parsed as ProfileMetadata).about === 'string') {\n metadata.about = (parsed as ProfileMetadata).about;\n }\n if (typeof (parsed as ProfileMetadata).picture === 'string') {\n metadata.picture = (parsed as ProfileMetadata).picture;\n }\n if (typeof (parsed as ProfileMetadata).website === 'string') {\n metadata.website = (parsed as ProfileMetadata).website;\n }\n return metadata;\n } catch (error) {\n console.error('Failed to parse profile metadata:', error);\n return null;\n }\n }\n\n private async enforceRateLimit(action: string, minIntervalMs: number): Promise<void> {\n const lastAt = this.lastActionAt.get(action) ?? 0;\n const now = Date.now();\n const waitMs = minIntervalMs - (now - lastAt);\n if (waitMs > 0) {\n await new Promise((resolve) => setTimeout(resolve, waitMs));\n }\n this.lastActionAt.set(action, Date.now());\n }\n}\n\n","import { useState } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { AuthState } from '../../types/auth';\nimport './Auth.css';\n\ninterface LoginButtonProps {\n authService: AuthService;\n setAuthenticated: AuthState['setAuthenticated'];\n setLoginError: AuthState['setLoginError'];\n onSuccess?: () => void;\n}\n\nexport function LoginButton({\n authService,\n setAuthenticated,\n setLoginError,\n onSuccess,\n}: LoginButtonProps) {\n const [isLoading, setIsLoading] = useState(false);\n\n const handleLogin = async () => {\n setIsLoading(true);\n setLoginError(null);\n\n try {\n if (!authService.hasKeyInfo()) {\n throw new Error('No account found. Please register first.');\n }\n\n const keyInfo = authService.getCurrentKeyInfo();\n if (!keyInfo) {\n throw new Error('Failed to load account information.');\n }\n\n await authService.getPublicKey();\n setAuthenticated(keyInfo);\n\n if (onSuccess) {\n onSuccess();\n }\n } catch (err) {\n console.error('Login error:', err);\n const errorMessage = err instanceof Error ? err.message : 'Failed to login';\n setLoginError(errorMessage);\n setIsLoading(false);\n }\n };\n\n return (\n <div className=\"login-button-container\">\n <button\n className=\"auth-button secondary\"\n onClick={handleLogin}\n disabled={isLoading}\n >\n {isLoading ? 'Logging in...' : 'Login'}\n </button>\n </div>\n );\n}\n\n","import { useState } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { AuthState } from '../../types/auth';\nimport './Auth.css';\n\ninterface RegistrationFlowProps {\n authService: AuthService;\n setAuthenticated: AuthState['setAuthenticated'];\n onSuccess?: () => void;\n}\n\nexport function RegistrationFlow({\n authService,\n setAuthenticated,\n onSuccess,\n}: RegistrationFlowProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n const [step, setStep] = useState<'info' | 'creating' | 'success'>('info');\n const [username, setUsername] = useState('');\n const maxUsernameLength = 100;\n\n const handleRegister = async () => {\n setIsLoading(true);\n setError(null);\n setStep('creating');\n\n try {\n authService.clearStoredKeyInfo();\n\n let credentialId: Uint8Array;\n const passkeyUsername = username.trim() || undefined;\n \n try {\n credentialId = await authService.createPasskey(passkeyUsername);\n } catch (passkeyError) {\n console.error('[RegistrationFlow] Passkey creation failed:', passkeyError);\n throw new Error('Unable to create passkey. Please try again.');\n }\n\n let keyInfo;\n try {\n keyInfo = await authService.createNostrKey(credentialId);\n } catch (nostrKeyError) {\n console.error('[RegistrationFlow] Failed to create Nostr key from passkey:', nostrKeyError);\n throw new Error('Unable to finalize account. Please try again.');\n }\n\n authService.setCurrentKeyInfo(keyInfo);\n setAuthenticated(keyInfo);\n setStep('success');\n\n if (onSuccess) {\n setTimeout(() => {\n onSuccess();\n }, 1500);\n }\n } catch (err) {\n console.error('[RegistrationFlow] Registration error:', err);\n setError('Registration failed. Please try again.');\n setStep('info');\n setIsLoading(false);\n }\n };\n\n return (\n <div className=\"auth-container\">\n <div className=\"auth-card\">\n <h1>Create Account</h1>\n <p className=\"auth-description\">\n Create a new decentralized Identity. Your identity will be securely derived from a passkey.\n </p>\n\n {step === 'info' && (\n <>\n <div className=\"auth-features\">\n <div className=\"feature-item\">\n <span className=\"feature-icon\">🔐</span>\n <span>Phishing-resistant authentication</span>\n </div>\n <div className=\"feature-item\">\n <span className=\"feature-icon\">📱</span>\n <span>Biometric authentication support</span>\n </div>\n <div className=\"feature-item\">\n <span className=\"feature-icon\">🌐</span>\n <span>Cross-device synchronization</span>\n </div>\n </div>\n\n <div className=\"username-section\">\n <label htmlFor=\"username\" className=\"username-label\">\n Name (Optional)\n </label>\n <input\n id=\"username\"\n type=\"text\"\n className=\"username-input\"\n placeholder=\"Enter a name for this passkey\"\n value={username}\n onChange={(e) => setUsername(e.target.value)}\n disabled={isLoading}\n maxLength={maxUsernameLength}\n />\n </div>\n\n {error && <div className=\"error-message\">{error}</div>}\n\n <button\n className=\"auth-button primary\"\n onClick={handleRegister}\n disabled={isLoading}\n >\n {isLoading ? 'Creating...' : 'Create Account'}\n </button>\n </>\n )}\n\n {step === 'creating' && (\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Creating your passkey...</p>\n <p className=\"loading-hint\">Please follow your browser's authentication prompt</p>\n <p className=\"loading-hint-small\">\n 💡 Using your system's native passkey manager (Touch ID, Face ID, Windows Hello, etc.)\n </p>\n </div>\n )}\n\n {step === 'success' && (\n <div className=\"success-state\">\n <div className=\"success-icon\">✓</div>\n <p>Account created successfully!</p>\n <p className=\"success-hint\">Redirecting to profile setup...</p>\n </div>\n )}\n </div>\n </div>\n );\n}\n\n","import DOMPurify from 'isomorphic-dompurify';\n\nconst sanitizeOptions = {\n ALLOWED_TAGS: [],\n ALLOWED_ATTR: [],\n};\n\nexport const sanitizeText = (value: string): string =>\n DOMPurify.sanitize(value, sanitizeOptions);\n","import { useState } from 'react';\nimport { useZxing } from 'react-zxing';\n\ninterface BarcodeScannerProps {\n /** Called when a code is successfully read */\n onDecode: (value: string) => void;\n /** Optional flag to hide the scanner */\n active?: boolean;\n}\n\nexport const BarcodeScanner = ({ onDecode, active = true }: BarcodeScannerProps) => {\n const [error, setError] = useState<string | null>(null);\n const { ref } = useZxing({\n onDecodeResult: (result) => {\n onDecode(result.getText());\n },\n onError: (e: unknown) => {\n const errorMessage = e instanceof Error ? e.message : 'Camera error';\n setError(errorMessage);\n },\n });\n\n if (!active) return null;\n\n return (\n <div style={{ position: 'relative', width: '100%', maxWidth: '400px' }}>\n <video\n ref={ref as React.LegacyRef<HTMLVideoElement>}\n style={{ width: '100%', borderRadius: '8px' }}\n playsInline\n muted\n />\n {error && (\n <p style={{ color: 'red', marginTop: '0.5rem' }}>\n {error}\n </p>\n )}\n </div>\n );\n};\n\n","import { useState, useEffect } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { RelayService } from '../../services/relay.service';\nimport type { AuthState } from '../../types/auth';\nimport type { ProfileMetadata, FollowEntry } from '../../types/nostr';\nimport { sanitizeText } from '../../utils/sanitize';\nimport { isValidHttpUrl } from '../../utils/validation';\nimport { BarcodeScanner } from './BarcodeScanner';\nimport './Membership.css';\n\ninterface ProfileWithPubkey extends ProfileMetadata {\n pubkey: string;\n}\n\ninterface MembershipPageProps {\n authService: AuthService;\n relayService: RelayService;\n publicKey: string | null;\n onUnauthenticated?: () => void;\n}\n\nconst TrustInfo = () => (\n <section className=\"trust-info\">\n <h2>Building Trust with Verifiable Relationship Credentials</h2>\n <p>\n Beyond the Personhood Credential, members can create <strong>Verifiable Relationship\n Credentials (VRCs)</strong>. A VRC is issued directly between two members – for example,\n by scanning a QR‑code at a meetup – and certifies a first‑hand trust link.\n </p>\n <p>\n Each VRC becomes a node in a decentralized trust graph. When you add a member, the\n system records a <strong>role</strong> tag that ties the new\n participant to your existing graph, enabling permissionless access to network‑state\n resources while keeping the underlying data private.\n </p>\n <p>\n The graph grows organically: trusted authorities issue PHCs, and members continuously\n enrich the network with peer‑generated VRCs, creating a resilient, scalable web of\n verified participants.\n </p>\n </section>\n);\n\nexport function MembershipPage({\n authService,\n relayService,\n publicKey,\n onUnauthenticated,\n}: MembershipPageProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [isSaving, setIsSaving] = useState(false);\n const [saveMessage, setSaveMessage] = useState<string | null>(null);\n const [searchQuery, setSearchQuery] = useState('');\n const [profiles, setProfiles] = useState<ProfileWithPubkey[]>([]);\n const [members, setMembers] = useState<FollowEntry[]>([]);\n const [memberProfiles, setMemberProfiles] = useState<Map<string, ProfileMetadata>>(new Map());\n const [showScanner, setShowScanner] = useState(false);\n\n useEffect(() => {\n if (!publicKey) {\n if (onUnauthenticated) {\n onUnauthenticated();\n }\n return;\n }\n\n loadFollowList();\n }, [publicKey]);\n\n useEffect(() => {\n if (members.length > 0) {\n loadMemberProfiles();\n }\n }, [members]);\n\n const handleDecoded = (decoded: string) => {\n setSearchQuery(decoded);\n setShowScanner(false);\n handleSearch();\n };\n\n const loadFollowList = async () => {\n if (!publicKey) return;\n\n setIsLoading(true);\n try {\n const followList = await relayService.fetchFollowList(publicKey);\n setMembers(followList);\n } catch (error) {\n console.error('Failed to load follow list:', error);\n setSaveMessage('Failed to load membership list');\n } finally {\n setIsLoading(false);\n }\n };\n\n const loadMemberProfiles = async () => {\n if (members.length === 0) return;\n\n try {\n const pubkeys = members.map((m) => m.pubkey);\n const profilesMap = await relayService.fetchMultipleProfiles(pubkeys);\n setMemberProfiles(profilesMap);\n } catch (error) {\n console.error('Failed to load member profiles:', error);\n }\n };\n\n const handleSearch = async () => {\n if (!searchQuery.trim()) {\n setIsLoading(true);\n try {\n const profilesMap = await relayService.queryProfiles([], 50);\n const profilesList: ProfileWithPubkey[] = [];\n profilesMap.forEach((profile, pubkey) => {\n profilesList.push({ ...profile, pubkey });\n });\n setProfiles(profilesList);\n } catch (error) {\n console.error('Failed to query profiles:', error);\n setSaveMessage('Failed to search profiles');\n } finally {\n setIsLoading(false);\n }\n return;\n }\n\n const trimmedQuery = searchQuery.trim();\n if (trimmedQuery.length !== 64 || !/^[0-9a-fA-F]+$/.test(trimmedQuery)) {\n setSaveMessage('Invalid pubkey format. Must be 64 hex characters.');\n setTimeout(() => setSaveMessage(null), 3000);\n return;\n }\n\n setIsLoading(true);\n try {\n const profilesMap = await relayService.queryProfiles([trimmedQuery], 1);\n const profilesList: ProfileWithPubkey[] = [];\n profilesMap.forEach((profile, pubkey) => {\n profilesList.push({ ...profile, pubkey });\n });\n setProfiles(profilesList);\n if (profilesList.length === 0) {\n setSaveMessage('No profile found for this pubkey');\n setTimeout(() => setSaveMessage(null), 3000);\n }\n } catch (error) {\n console.error('Failed to query profile:', error);\n setSaveMessage('Failed to search profile');\n } finally {\n setIsLoading(false);\n }\n };\n\n const handleAddMember = async (pubkey: string) => {\n if (!publicKey) return;\n\n if (members.some((m) => m.pubkey === pubkey)) {\n setSaveMessage('User is already a member');\n setTimeout(() => setSaveMessage(null), 3000);\n return;\n }\n\n setIsSaving(true);\n try {\n const newMembers: FollowEntry[] = [\n ...members,\n {\n pubkey,\n },\n ];\n\n await relayService.publishFollowList(publicKey, newMembers, (event) =>\n authService.signEvent(event)\n );\n\n setMembers(newMembers);\n setSaveMessage('Member added successfully!');\n setTimeout(() => setSaveMessage(null), 3000);\n } catch (error) {\n console.error('Failed to add member:', error);\n setSaveMessage('Failed to add member. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const handleRemoveMember = async (pubkey: string) => {\n if (!publicKey) return;\n\n setIsSaving(true);\n try {\n const newMembers = members.filter((m) => m.pubkey !== pubkey);\n\n await relayService.publishFollowList(publicKey, newMembers, (event) =>\n authService.signEvent(event)\n );\n\n setMembers(newMembers);\n setSaveMessage('Member removed successfully!');\n setTimeout(() => setSaveMessage(null), 3000);\n } catch (error) {\n console.error('Failed to remove member:', error);\n setSaveMessage('Failed to remove member. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const getProfileDisplayName = (profile: ProfileMetadata, pubkey: string): string => {\n return profile.display_name || profile.name || pubkey.slice(0, 16) + '...';\n };\n\n const formatPubkey = (pubkey: string): string => {\n return `${pubkey.slice(0, 8)}...${pubkey.slice(-8)}`;\n };\n\n if (isLoading && members.length === 0) {\n return (\n <div className=\"membership-container\">\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Loading membership list...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"membership-container\">\n <div className=\"membership-card\">\n <h1>Membership Management</h1>\n <TrustInfo />\n <p className=\"membership-description\">\n Query applicants and manage the membership list.\n </p>\n\n {saveMessage && (\n <div\n className={`save-message ${\n saveMessage.includes('Error') || saveMessage.includes('Failed')\n ? 'error'\n : 'success'\n }`}\n >\n {saveMessage}\n </div>\n )}\n\n <div className=\"search-section\">\n <h2>Add Member</h2>\n <div className=\"search-form\">\n <input\n type=\"text\"\n value={searchQuery}\n onChange={(e) => setSearchQuery(e.target.value)}\n onKeyPress={(e) => e.key === 'Enter' && handleSearch()}\n placeholder=\"Enter pubkey or leave empty for recent profiles\"\n className=\"search-input\"\n disabled={isLoading}\n />\n <button\n onClick={handleSearch}\n className=\"search-button\"\n disabled={isLoading || isSaving}\n >\n {isLoading ? 'Searching...' : 'Search'}\n </button>\n <button\n onClick={() => setShowScanner((prev) => !prev)}\n className=\"scanner-toggle\"\n disabled={isLoading || isSaving}\n >\n {showScanner ? 'Close Scanner' : 'Scan QR'}\n </button>\n </div>\n\n {showScanner && (\n <div style={{ marginTop: '1rem' }}>\n <BarcodeScanner onDecode={handleDecoded} active={true} />\n <p style={{ fontSize: '0.85rem', marginTop: '0.5rem' }}>\n Point at QR‑code containing a pubkey\n </p>\n </div>\n )}\n\n {profiles.length > 0 && (\n <div className=\"profiles-list\">\n <h3>Search Results</h3>\n {profiles.map((profile) => {\n const isMember = members.some((m) => m.pubkey === profile.pubkey);\n return (\n <div key={profile.pubkey} className=\"profile-item\">\n <div className=\"profile-info\">\n {profile.picture && isValidHttpUrl(profile.picture) && (\n <img\n src={profile.picture}\n alt={sanitizeText(getProfileDisplayName(profile, profile.pubkey))}\n className=\"profile-avatar\"\n loading=\"lazy\"\n referrerPolicy=\"no-referrer\"\n onError={(event) => {\n event.currentTarget.style.display = 'none';\n }}\n />\n )}\n <div className=\"profile-details\">\n <div className=\"profile-name\">\n {sanitizeText(getProfileDisplayName(profile, profile.pubkey))}\n </div>\n <div className=\"profile-pubkey\">{formatPubkey(profile.pubkey)}</div>\n {profile.about && (\n <div className=\"profile-about\">{sanitizeText(profile.about)}</div>\n )}\n </div>\n </div>\n <button\n onClick={() =>\n isMember\n ? handleRemoveMember(profile.pubkey)\n : handleAddMember(profile.pubkey)\n }\n className={`member-button ${isMember ? 'remove' : 'add'}`}\n disabled={isSaving}\n >\n {isMember ? 'Remove' : 'Add Member'}\n </button>\n </div>\n );\n })}\n </div>\n )}\n </div>\n\n <div className=\"members-section\">\n <h2>Current Members ({members.length})</h2>\n {members.length === 0 ? (\n <p className=\"empty-message\">No members yet. Add members from search results above.</p>\n ) : (\n <div className=\"members-list\">\n {members.map((member) => {\n const profile = memberProfiles.get(member.pubkey);\n return (\n <div key={member.pubkey} className=\"member-item\">\n <div className=\"profile-info\">\n {profile?.picture && isValidHttpUrl(profile.picture) && (\n <img\n src={profile.picture}\n alt={sanitizeText(getProfileDisplayName(profile || {}, member.pubkey))}\n className=\"profile-avatar\"\n loading=\"lazy\"\n referrerPolicy=\"no-referrer\"\n onError={(event) => {\n event.currentTarget.style.display = 'none';\n }}\n />\n )}\n <div className=\"profile-details\">\n <div className=\"profile-name\">\n {profile\n ? sanitizeText(getProfileDisplayName(profile, member.pubkey))\n : formatPubkey(member.pubkey)}\n </div>\n <div className=\"profile-pubkey\">{formatPubkey(member.pubkey)}</div>\n {profile?.about && (\n <div className=\"profile-about\">{sanitizeText(profile.about)}</div>\n )}\n {member.petname && (\n <div className=\"profile-petname\">Name: {sanitizeText(member.petname)}</div>\n )}\n </div>\n </div>\n <button\n onClick={() => handleRemoveMember(member.pubkey)}\n className=\"member-button remove\"\n disabled={isSaving}\n >\n Remove\n </button>\n </div>\n );\n })}\n </div>\n )}\n </div>\n </div>\n </div>\n );\n}\n\n","import { useState, useEffect } from 'react';\nimport type { AuthService } from '../../services/auth.service';\nimport type { RelayService } from '../../services/relay.service';\nimport type { AuthState } from '../../types/auth';\nimport type { ProfileMetadata } from '../../types/nostr';\nimport { sanitizeText } from '../../utils/sanitize';\nimport { isValidPubkey, isValidRoleTag } from '../../utils/validation';\nimport './Profile.css';\n\ninterface ProfilePageProps {\n authService: AuthService;\n relayService: RelayService;\n publicKey: string | null;\n onUnauthenticated?: () => void;\n onSuccess?: () => void;\n onRoleSuggestion?: (about: string) => Promise<string | null>;\n}\n\nconst PersonhoodInfo = () => (\n <section className=\"personhood-info\">\n <p>\n Network‑state members receive a <strong>Personhood Credential (PHC)</strong> from a trusted authority.\n The PHC attests that the holder is a unique, real individual. Because the credential lives\n in your passport, you can present a <em>zero‑knowledge proof</em> that you are verified.\n </p>\n <p>\n In this form we compare the name you enter with the name disclosed by your verified\n passport proof. If the two match, the profile will be saved as your business card credential together with a\n passport tag that references the PHC's unique identifier.\n </p>\n </section>\n);\n\nconst MAX_NAME_LENGTH = 100;\nconst MAX_ABOUT_LENGTH = 1000;\nconst MAX_URL_LENGTH = 2048;\n\nexport function ProfilePage({\n authService,\n relayService,\n publicKey,\n onUnauthenticated,\n onSuccess,\n onRoleSuggestion,\n}: ProfilePageProps) {\n const [isLoading, setIsLoading] = useState(false);\n const [isSaving, setIsSaving] = useState(false);\n const [saveMessage, setSaveMessage] = useState<string | null>(null);\n const [suggestedRole, setSuggestedRole] = useState<string | null>(null);\n const [isGettingRole, setIsGettingRole] = useState(false);\n const [formData, setFormData] = useState<ProfileMetadata>({\n name: '',\n display_name: '',\n about: '',\n picture: '',\n website: '',\n });\n\n useEffect(() => {\n if (!publicKey) {\n if (onUnauthenticated) {\n onUnauthenticated();\n }\n return;\n }\n\n loadProfile();\n }, [publicKey]);\n\n const loadProfile = async () => {\n if (!publicKey) return;\n\n setIsLoading(true);\n try {\n const [profile, roleTag] = await Promise.all([\n relayService.fetchProfile(publicKey),\n relayService.fetchProfileRoleTag(publicKey),\n ]);\n \n if (profile) {\n setFormData({\n name: sanitizeText(profile.name || ''),\n display_name: sanitizeText(profile.display_name || ''),\n about: sanitizeText(profile.about || ''),\n picture: profile.picture || '',\n website: profile.website || '',\n });\n }\n \n if (roleTag) {\n setSuggestedRole(roleTag);\n }\n } catch (error) {\n console.error('Failed to load profile:', error);\n } finally {\n setIsLoading(false);\n }\n };\n\n const handleSubmit = async (e: React.FormEvent) => {\n e.preventDefault();\n if (!publicKey) return;\n\n setIsSaving(true);\n setSaveMessage(null);\n\n try {\n const tags: string[][] = [];\n if (!isValidPubkey(publicKey)) {\n throw new Error('Invalid public key format');\n }\n\n // Suggest role based on \"about\" field if it has content and callback provided\n if (formData.about && formData.about.trim().length > 0 && onRoleSuggestion) {\n setIsGettingRole(true);\n try {\n const roleSuggestion = await onRoleSuggestion(formData.about);\n const candidate = roleSuggestion ? roleSuggestion.trim() : '';\n if (candidate && isValidRoleTag(candidate)) {\n tags.push(['role', candidate]);\n setSuggestedRole(candidate);\n } else {\n setSuggestedRole(null);\n }\n } catch (error) {\n console.error('Failed to get role suggestion:', error);\n setSuggestedRole(null);\n } finally {\n setIsGettingRole(false);\n }\n } else {\n setSuggestedRole(null);\n }\n\n const profileEvent = {\n kind: 0,\n content: JSON.stringify({\n ...formData,\n name: sanitizeText(formData.name || ''),\n display_name: sanitizeText(formData.display_name || ''),\n about: sanitizeText(formData.about || ''),\n }),\n created_at: Math.floor(Date.now() / 1000),\n tags,\n };\n\n const follows = {\n kind: 3,\n content: \"\",\n created_at: Math.floor(Date.now() / 1000),\n tags: [\n (() => {\n const followTag: string[] = ['p', publicKey];\n const relayUrl = relayService.getRelays()[0];\n if (relayUrl) {\n followTag.push(relayUrl);\n }\n if (formData.name) {\n followTag.push(formData.name);\n }\n return followTag;\n })(),\n ],\n };\n\n const signedProfile = await authService.signEvent(profileEvent);\n const signedFollows = await authService.signEvent(follows);\n\n await relayService.publishEvent(signedProfile);\n await relayService.publishEvent(signedFollows);\n\n setSaveMessage('Profile saved successfully!');\n setTimeout(() => {\n setSaveMessage(null);\n }, 3000);\n\n if (onSuccess) {\n onSuccess();\n }\n } catch (error) {\n console.error('Failed to save profile:', error);\n setSaveMessage('Failed to save profile. Please try again.');\n } finally {\n setIsSaving(false);\n }\n };\n\n const handleChange = (\n e: React.ChangeEvent<HTMLInputElement | HTMLTextAreaElement>\n ) => {\n const { name, value } = e.target;\n setFormData((prev) => ({\n ...prev,\n [name]: value,\n }));\n };\n\n if (isLoading) {\n return (\n <div className=\"profile-container\">\n <div className=\"loading-state\">\n <div className=\"spinner\"></div>\n <p>Loading profile...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"profile-container\">\n <div className=\"profile-card\">\n <h1>Profile Setup</h1>\n <PersonhoodInfo />\n\n <form onSubmit={handleSubmit} className=\"profile-form\">\n <div className=\"form-group\">\n <label htmlFor=\"name\">First Name</label>\n <input\n type=\"text\"\n id=\"name\"\n name=\"name\"\n value={formData.name || ''}\n onChange={handleChange}\n placeholder=\"Your username\"\n maxLength={MAX_NAME_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"display_name\">Last Name</label>\n <input\n type=\"text\"\n id=\"display_name\"\n name=\"display_name\"\n value={formData.display_name || ''}\n onChange={handleChange}\n placeholder=\"Your Last Name\"\n maxLength={MAX_NAME_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"about\">About</label>\n <textarea\n id=\"about\"\n name=\"about\"\n value={formData.about || ''}\n onChange={handleChange}\n placeholder=\"Tell us about yourself\"\n rows={4}\n maxLength={MAX_ABOUT_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"picture\">Profile Picture URL</label>\n <input\n type=\"url\"\n id=\"picture\"\n name=\"picture\"\n value={formData.picture || ''}\n onChange={handleChange}\n placeholder=\"https://example.com/avatar.jpg\"\n maxLength={MAX_URL_LENGTH}\n />\n </div>\n\n <div className=\"form-group\">\n <label htmlFor=\"website\">LinkedIn</label>\n <input\n type=\"url\"\n id=\"website\"\n name=\"website\"\n value={formData.website || ''}\n onChange={handleChange}\n placeholder=\"https://linkedin.com/example\"\n maxLength={MAX_URL_LENGTH}\n />\n </div>\n\n {saveMessage && (\n <div className={`save-message ${saveMessage.includes('Error') ? 'error' : 'success'}`}>\n {saveMessage}\n </div>\n )}\n\n {(isGettingRole || suggestedRole) && (\n <div className=\"role-tag-container\">\n {isGettingRole ? (\n <>\n <div className=\"role-tag-label\">Getting AI suggestion...</div>\n <div className=\"role-tag-loading\">\n <div className=\"role-tag-spinner\"></div>\n </div>\n </>\n ) : suggestedRole ? (\n <>\n <div className=\"role-tag-label\">AI Suggested Role:</div>\n <div className=\"role-tag\">\n {sanitizeText(suggestedRole)}\n </div>\n </>\n ) : null}\n </div>\n )}\n\n <div className=\"form-actions\">\n <button type=\"submit\" className=\"save-button\" disabled={isSaving}>\n {isSaving ? 'Saving...' : 'Save Profile'}\n </button>\n </div>\n </form>\n </div>\n </div>\n );\n}\n\n","import { useEffect } from 'react';\nimport type { AuthService } from '../services/auth.service';\nimport type { AuthState } from '../types/auth';\n\n/**\n * Hook to initialize auth state on app load\n */\nexport function useAuthInit(\n authService: AuthService,\n setAuthenticated: AuthState['setAuthenticated']\n): void {\n useEffect(() => {\n if (authService.hasKeyInfo()) {\n const keyInfo = authService.getCurrentKeyInfo();\n if (keyInfo) {\n setAuthenticated(keyInfo);\n }\n }\n }, [authService, setAuthenticated]);\n}\n\n","import { create } from 'zustand';\nimport type { NostrKeyInfo } from '../utils';\nimport type { AuthState } from '../types/auth';\n\nexport const createAuthStore = () => {\n return create<AuthState>((set) => ({\n isAuthenticated: false,\n publicKey: null,\n keyInfo: null,\n loginError: null,\n setAuthenticated: (keyInfo: NostrKeyInfo | null) => {\n set({\n isAuthenticated: !!keyInfo,\n publicKey: keyInfo?.pubkey || null,\n keyInfo,\n loginError: null,\n });\n },\n setLoginError: (error: string | null) => {\n set({ loginError: error });\n },\n logout: () => {\n set({\n isAuthenticated: false,\n publicKey: null,\n keyInfo: null,\n loginError: null,\n });\n },\n }));\n};\n\n// Default singleton instance\nexport const useAuthStore = createAuthStore();\n\n"],"mappings":"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"}
package/dist/index.d.cts CHANGED
@@ -52,6 +52,7 @@ interface AuthServiceConfig {
52
52
  rpName?: string;
53
53
  storageKey?: string;
54
54
  cacheTimeoutMs?: number;
55
+ cacheOnCreation?: boolean;
55
56
  }
56
57
  /**
57
58
  * Relay service configuration
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.cts","names":[],"sources":["../src/utils/types.ts","../src/types/auth.ts","../src/services/auth.service.ts","../src/types/nostr.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/components/membership/MembershipPage.tsx","../src/components/membership/BarcodeScanner.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":[],"mappings":"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"}
1
+ {"version":3,"file":"index.d.cts","names":[],"sources":["../src/utils/types.ts","../src/types/auth.ts","../src/services/auth.service.ts","../src/types/nostr.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/components/membership/MembershipPage.tsx","../src/components/membership/BarcodeScanner.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":[],"mappings":"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"}
package/dist/index.d.mts CHANGED
@@ -52,6 +52,7 @@ interface AuthServiceConfig {
52
52
  rpName?: string;
53
53
  storageKey?: string;
54
54
  cacheTimeoutMs?: number;
55
+ cacheOnCreation?: boolean;
55
56
  }
56
57
  /**
57
58
  * Relay service configuration
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.mts","names":[],"sources":["../src/utils/types.ts","../src/types/auth.ts","../src/services/auth.service.ts","../src/types/nostr.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/components/membership/MembershipPage.tsx","../src/components/membership/BarcodeScanner.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":[],"mappings":"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"}
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../src/utils/types.ts","../src/types/auth.ts","../src/services/auth.service.ts","../src/types/nostr.ts","../src/services/relay.service.ts","../src/components/auth/LoginButton.tsx","../src/components/auth/RegistrationFlow.tsx","../src/components/membership/MembershipPage.tsx","../src/components/membership/BarcodeScanner.tsx","../src/components/profile/ProfilePage.tsx","../src/hooks/useAuth.ts","../src/store/authStore.ts"],"sourcesContent":[],"mappings":";;;;;;;;;;;AAQA;AAcA;UAdiB,YAAA;;;ECHA,UAAA,CAAA,EAAS,MAAA;EAaT,IAAA,EAAA,MAAA;EAWA,IAAA,CAAA,EAAA,MAAA,EAAA,EAAA;;;;ACtBjB;;;;AAiFsC,UFlErB,YAAA,CEkEqB;EAAqB,YAAA,EAAA,MAAA;EAAR,MAAA,EAAA,MAAA;EAQ3B,IAAA,EAAA,MAAA;EAQC,QAAA,CAAA,EAAA,MAAA;;;;;;;UDnGR,SAAA;EDGA,eAAU,EAAA,OAAA;EAcV,SAAA,EAAA,MAAY,GAAA,IAAA;WCdlB;;8BAEmB;EALb,aAAS,EAAA,CAAA,KAAA,EAAA,MAGf,GAAA,IAAA,EAEmB,GAAA,IAAA;EAQb,MAAA,EAAA,GAAA,GAAA,IAAA;AAWjB;;;;ACtBa,UDWI,iBAAA,CCXO;EAIF,IAAA,CAAA,EAAA,MAAA;EAyC4B,MAAA,CAAA,EAAA,MAAA;EAAR,UAAA,CAAA,EAAA,MAAA;EAoCJ,cAAA,CAAA,EAAA,MAAA;EAAqB,eAAA,CAAA,EAAA,OAAA;;;;;AAgBrB,UD3ErB,kBAAA,CC2EqB;EAQf,SAAA,CAAA,EAAA,MAAA,EAAA;;;;;;;AFxGvB;AAciB,cEfJ,WAAA,CFegB;;;uBEXP;EDNL,QAAA,gBAGN;EAUM;AAWjB;;;;ACtBA;;;EA6C0C,aAAA,CAAA,QAAA,CAAA,EAAA,MAAA,CAAA,EAAA,OAAA,CAAQ,UAAR,CAAA;EAoCJ;;;EAQd,cAAA,CAAA,YAAA,CAAA,EARc,UAQd,CAAA,EAR2B,OAQ3B,CARmC,YAQnC,CAAA;EAQC;;;EAQF,YAAA,CAAA,CAAA,EAhBC,OAgBD,CAAA,MAAA,CAAA;EAQM;;;mBAhBJ,eAAa,QAAQ;;;ACrG9C;EAaiB,iBAAA,CAAA,CAAe,EDgGT,YChGS,GAAA,IAAA;EAYf;;;6BD4FY;EE/GhB;;;EA4Ce,UAAA,CAAA,CAAA,EAAA,OAAA;EAA+B;;;EA+Fd,kBAAA,CAAA,CAAA,EAAA,IAAA;EAyDI;;;EA2DS,cAAA,CAAA,CAAA,EFxHhC,OEwHgC,CAAA,OAAA,CAAA;;;;;;;UDrQzC,UAAA;;EHKA,MAAA,CAAA,EAAA,MAAU;EAcV,UAAA,CAAA,EAAA,MAAY;;;;ECjBZ,GAAA,CAAA,EAAA,MAAS;AAa1B;AAWA;;;UEbiB,eAAA;EDTJ,IAAA,CAAA,EAAA,MAAA;EAIS,YAAA,CAAA,EAAA,MAAA;EAyC4B,KAAA,CAAA,EAAA,MAAA;EAAR,OAAA,CAAA,EAAA,MAAA;EAoCJ,OAAA,CAAA,EAAA,MAAA;EAAqB,CAAA,GAAA,EAAA,MAAA,CAAA,EAAA,OAAA;;;;;AAgBrB,UC5ErB,WAAA,CD4EqB;EAQf,MAAA,EAAA,MAAA;EAQM,KAAA,CAAA,EAAA,MAAA;EAwBH,OAAA,CAAA,EAAA,MAAA;;;;;AFxI1B;AAcA;cIba,YAAA;;;EHJI,QAAA,aAAS;EAaT,iBAAA,qBAAiB;EAWjB,iBAAA,yBAAkB;;;uBGXb;EFXT;;;EA6C6B,UAAA,CAAA,UAAA,EE3BjB,UF2BiB,CAAA,EAAA,IAAA;EAoCJ;;;EAQd,SAAA,CAAA,IAAA,EAAA,MAAA,EAAA,CAAA,EAAA,IAAA;EAQC;;;EAQF,SAAA,CAAA,CAAA,EAAA,MAAA,EAAA;EAQM;;;sBEnED,iCAA+B;;;ADlD3D;EAaiB,YAAA,CAAA,MAAe,EAAA,MAAA,CAAA,ECgFM,ODhFN,CCgFc,eDhFd,GAAA,IAAA,CAAA;EAYf;;;uCCwH4B;EA3IhC;;;EA4Ce,eAAA,CAAA,MAAA,EAAA,MAAA,CAAA,EAwJa,OAxJb,CAwJqB,WAxJrB,EAAA,CAAA;EAA+B;;;EA+Fd,qBAAA,CAAA,OAAA,EAAA,MAAA,EAAA,CAAA,EAoHK,OApHL,CAoHa,GApHb,CAAA,MAAA,EAoHyB,eApHzB,CAAA,CAAA;EAyDI;;;;EA2DC,aAAA,CAAA,OAAA,CAAA,EAAA,MAAA,EAAA,EAAA,KAAA,CAAA,EAAA,MAAA,CAAA,EAuDU,OAvDV,CAuDkB,GAvDlB,CAAA,MAAA,EAuD8B,eAvD9B,CAAA,CAAA;EAuD8B;;;EAwEhE,iBAAA,CAAA,MAAA,EAAA,MAAA,EAAA,UAAA,EAAA,WAAA,EAAA,EAAA,SAAA,EAAA,CAAA,KAAA,EACO,UADP,EAAA,GACsB,OADtB,CAC8B,UAD9B,CAAA,CAAA,EAEX,OAFW,CAAA,OAAA,CAAA;EACO,QAAA,iBAAA;EAAuB,QAAA,oBAAA;EAAR,QAAA,gBAAA;;;;UCnY5B,gBAAA;eACK;ELEE,gBAAU,EKDP,SLCO,CAAA,kBAAA,CAAA;EAcV,aAAA,EKdA,SLcY,CAAA,eAAA,CAAA;;;iBKVb,WAAA;;;;;GAKb,mBAAgB,kBAAA,CAAA,GAAA,CAAA;;;UCZT,qBAAA;eACK;ENEE,gBAAU,EMDP,SNCO,CAAA,kBAAA,CAAA;EAcV,SAAA,CAAA,EAAA,GAAY,GAAA,IAAA;;iBMXb,gBAAA;;;;GAIb,wBAAqB,kBAAA,CAAA,GAAA,CAAA;;;UCDd,mBAAA;eACK;EPPE,YAAA,EOQD,YPRW;EAcV,SAAA,EAAA,MAAY,GAAA,IAAA;;;iBOqBb,cAAA;;;;;GAKb,sBAAmB,kBAAA,CAAA,GAAA,CAAA;;;UC7CZ,mBAAA;;;;;ARKV;AAciB,cQZJ,cRYgB,EAAA,CAAA;EAAA,QAAA;EAAA;AAAA,CAAA,EQZ+B,mBRY/B,EAAA,GQZkD,kBAAA,CAAA,GAAA,CAAA,OAAA,GRYlD,IAAA;;;USbnB,gBAAA;eACK;ETFE,YAAA,ESGD,YTHW;EAcV,SAAA,EAAA,MAAY,GAAA,IAAA;;;wCSPW;ARVxC;AAaiB,iBQmBD,WAAA,CRnBkB;EAAA,WAAA;EAAA,YAAA;EAAA,SAAA;EAAA,iBAAA;EAAA,SAAA;EAAA;AAAA,CAAA,EQ0B/B,gBR1B+B,CAAA,EQ0Bf,kBAAA,CAAA,GAAA,CAAA,OR1Be;;;;;;ADVjB,iBUDD,WAAA,CVCW,WAAA,EUAZ,WVAY,EAAA,gBAAA,EUCP,SVDO,CAAA,kBAAA,CAAA,CAAA,EAAA,IAAA;;;cWJd,uBAAe,QAAA,CAAA,cAAA,QAAA,CAAA,SAAA;cA6Bf,cAAY,QAAA,CAAA,cAAA,QAAA,CAAA,SAAA"}
package/dist/index.mjs CHANGED
@@ -1,4 +1,4 @@
1
- import { t as NosskeyManager } from "./utils-CV_lg5Ir.mjs";
1
+ import { t as NosskeyManager } from "./utils-DrZ-SwZX.mjs";
2
2
  import { useEffect, useState } from "react";
3
3
  import { Fragment, jsx, jsxs } from "react/jsx-runtime";
4
4
  import DOMPurify from "isomorphic-dompurify";
@@ -17,7 +17,8 @@ var AuthService = class {
17
17
  rpId: config.rpId || (typeof window !== "undefined" ? window.location.hostname.replace(/^www\./, "") : "localhost"),
18
18
  rpName: config.rpName || this.getDefaultRpName(),
19
19
  storageKey: config.storageKey || "nsauth_keyinfo",
20
- cacheTimeoutMs: config.cacheTimeoutMs || 1800 * 1e3
20
+ cacheTimeoutMs: config.cacheTimeoutMs || 1800 * 1e3,
21
+ cacheOnCreation: config.cacheOnCreation !== void 0 ? config.cacheOnCreation : true
21
22
  };
22
23
  }
23
24
  getDefaultRpName() {
@@ -33,7 +34,8 @@ var AuthService = class {
33
34
  if (!this.manager) this.manager = new NosskeyManager({
34
35
  cacheOptions: {
35
36
  enabled: true,
36
- timeoutMs: this.config.cacheTimeoutMs
37
+ timeoutMs: this.config.cacheTimeoutMs,
38
+ cacheOnCreation: this.config.cacheOnCreation
37
39
  },
38
40
  storageOptions: {
39
41
  enabled: true,
@@ -116,7 +118,7 @@ var AuthService = class {
116
118
  * Check if PRF is supported
117
119
  */
118
120
  async isPrfSupported() {
119
- const { isPrfSupported } = await import("./utils-BVvtCpMQ.mjs");
121
+ const { isPrfSupported } = await import("./utils-CIdi100K.mjs");
120
122
  return await isPrfSupported();
121
123
  }
122
124
  };