@growsober/sdk 1.0.1 → 1.0.2

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.
@@ -7,7 +7,7 @@
7
7
  * @module api/mutations/auth
8
8
  */
9
9
  import { UseMutationOptions, UseMutationResult } from '@tanstack/react-query';
10
- import type { RegisterRequest, LoginRequest, RefreshTokenRequest, FirebaseAuthRequest, AuthResponse, TokenResponse } from '../types';
10
+ import type { RegisterRequest, LoginRequest, RefreshTokenRequest, FirebaseAuthRequest, AuthResponse, TokenResponse, SendOtpRequest, OtpSentResponse, VerifyOtpRequest } from '../types';
11
11
  /**
12
12
  * Register a new user account
13
13
  *
@@ -265,3 +265,85 @@ export declare function useRefreshAuthToken(options?: Omit<UseMutationOptions<To
265
265
  * @returns TanStack Query mutation result
266
266
  */
267
267
  export declare function useFirebaseAuth(options?: Omit<UseMutationOptions<AuthResponse, Error, FirebaseAuthRequest>, 'mutationFn'>): UseMutationResult<AuthResponse, Error, FirebaseAuthRequest>;
268
+ /**
269
+ * Response type for verify OTP including isNewUser flag
270
+ */
271
+ export type VerifyOtpResponse = AuthResponse & {
272
+ isNewUser: boolean;
273
+ };
274
+ /**
275
+ * Send OTP to phone number
276
+ *
277
+ * @description
278
+ * Sends a verification code to the specified phone number via SMS or voice call.
279
+ * Used for passwordless authentication.
280
+ *
281
+ * @endpoint POST /auth/phone/send-otp
282
+ *
283
+ * @example
284
+ * ```tsx
285
+ * import { useSendOtp } from '@growsober/sdk';
286
+ *
287
+ * function PhoneInputScreen() {
288
+ * const { mutate: sendOtp, isPending, error } = useSendOtp({
289
+ * onSuccess: (data) => {
290
+ * console.log('OTP sent to:', data.phone);
291
+ * navigation.navigate('VerifyOtp', { phone });
292
+ * },
293
+ * onError: (error) => {
294
+ * Alert.alert('Error', error.message);
295
+ * },
296
+ * });
297
+ *
298
+ * const handleSend = () => {
299
+ * sendOtp({ phone: '+1234567890', channel: 'sms' });
300
+ * };
301
+ *
302
+ * return <Button onPress={handleSend} disabled={isPending} />;
303
+ * }
304
+ * ```
305
+ *
306
+ * @param options - TanStack Query mutation options
307
+ * @returns TanStack Query mutation result
308
+ */
309
+ export declare function useSendOtp(options?: Omit<UseMutationOptions<OtpSentResponse, Error, SendOtpRequest>, 'mutationFn'>): UseMutationResult<OtpSentResponse, Error, SendOtpRequest>;
310
+ /**
311
+ * Verify OTP and authenticate
312
+ *
313
+ * @description
314
+ * Verifies the OTP code sent to the phone number and authenticates the user.
315
+ * If the user doesn't exist, a new account is created automatically.
316
+ * Returns authentication tokens and user information.
317
+ *
318
+ * @endpoint POST /auth/phone/verify-otp
319
+ *
320
+ * @example
321
+ * ```tsx
322
+ * import { useVerifyOtp } from '@growsober/sdk';
323
+ *
324
+ * function VerifyOtpScreen({ phone }) {
325
+ * const { mutate: verifyOtp, isPending, error } = useVerifyOtp({
326
+ * onSuccess: async (data) => {
327
+ * await SecureStore.setItemAsync('accessToken', data.accessToken);
328
+ * await SecureStore.setItemAsync('refreshToken', data.refreshToken);
329
+ *
330
+ * if (data.isNewUser) {
331
+ * navigation.navigate('Onboarding');
332
+ * } else {
333
+ * navigation.navigate('Home');
334
+ * }
335
+ * },
336
+ * });
337
+ *
338
+ * const handleVerify = (code: string) => {
339
+ * verifyOtp({ phone, code });
340
+ * };
341
+ *
342
+ * return <OtpInput onComplete={handleVerify} disabled={isPending} />;
343
+ * }
344
+ * ```
345
+ *
346
+ * @param options - TanStack Query mutation options
347
+ * @returns TanStack Query mutation result
348
+ */
349
+ export declare function useVerifyOtp(options?: Omit<UseMutationOptions<VerifyOtpResponse, Error, VerifyOtpRequest>, 'mutationFn'>): UseMutationResult<VerifyOtpResponse, Error, VerifyOtpRequest>;
@@ -12,6 +12,8 @@ exports.useRegister = useRegister;
12
12
  exports.useLogin = useLogin;
13
13
  exports.useRefreshAuthToken = useRefreshAuthToken;
14
14
  exports.useFirebaseAuth = useFirebaseAuth;
15
+ exports.useSendOtp = useSendOtp;
16
+ exports.useVerifyOtp = useVerifyOtp;
15
17
  const react_query_1 = require("@tanstack/react-query");
16
18
  const client_1 = require("../client");
17
19
  const users_1 = require("../queries/users");
@@ -329,4 +331,103 @@ function useFirebaseAuth(options) {
329
331
  ...options,
330
332
  });
331
333
  }
332
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/api/mutations/auth.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Auth Mutation Hooks\n *\n * TanStack Query mutation hooks for authentication-related write operations.\n * These hooks handle user registration, login, token refresh, and Firebase authentication.\n *\n * @module api/mutations/auth\n */\n\nimport { useMutation, UseMutationOptions, UseMutationResult, useQueryClient } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport { userKeys } from '../queries/users';\nimport type {\n  RegisterRequest,\n  LoginRequest,\n  RefreshTokenRequest,\n  FirebaseAuthRequest,\n  AuthResponse,\n  TokenResponse,\n} from '../types';\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Register a new user account\n *\n * @description\n * Creates a new user account with email/phone and password.\n * Returns authentication tokens and user information upon successful registration.\n *\n * @endpoint POST /auth/register\n *\n * @example\n * ```tsx\n * import { useRegister } from '@growsober/sdk';\n *\n * function RegisterForm() {\n *   const { mutate: register, isPending, error } = useRegister({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Onboarding');\n *     },\n *     onError: (error) => {\n *       Alert.alert('Registration failed', error.message);\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     register({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *       name: 'John Doe',\n *     });\n *   };\n *\n *   return <Button onPress={handleSubmit} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRegister(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, RegisterRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, RegisterRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: RegisterRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/register', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Login with email/phone and password\n *\n * @description\n * Authenticates an existing user with their credentials.\n * Returns authentication tokens and user information upon successful login.\n *\n * @endpoint POST /auth/login\n *\n * @example\n * ```tsx\n * import { useLogin } from '@growsober/sdk';\n *\n * function LoginForm() {\n *   const { mutate: login, isPending, error } = useLogin({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Home');\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     login({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *     });\n *   };\n *\n *   return (\n *     <form onSubmit={handleSubmit}>\n *       <input type=\"email\" name=\"email\" />\n *       <input type=\"password\" name=\"password\" />\n *       <button type=\"submit\" disabled={isPending}>\n *         {isPending ? 'Logging in...' : 'Login'}\n *       </button>\n *       {error && <p className=\"error\">{error.message}</p>}\n *     </form>\n *   );\n * }\n * ```\n *\n * @example\n * Login with phone number:\n * ```tsx\n * login({\n *   phone: '+1234567890',\n *   password: 'SecurePassword123!',\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useLogin(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, LoginRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, LoginRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: LoginRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/login', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Refresh access token using refresh token\n *\n * @description\n * Obtains a new access token using a valid refresh token.\n * Should be called when the access token expires.\n *\n * @endpoint POST /auth/refresh\n *\n * @example\n * ```tsx\n * import { useRefreshAuthToken } from '@growsober/sdk';\n *\n * function useTokenRefresh() {\n *   const { mutateAsync: refreshToken } = useRefreshAuthToken();\n *\n *   const handleTokenExpired = async () => {\n *     const storedRefreshToken = await SecureStore.getItemAsync('refreshToken');\n *\n *     if (!storedRefreshToken) {\n *       navigation.navigate('Login');\n *       return;\n *     }\n *\n *     try {\n *       const { accessToken, refreshToken: newRefreshToken } = await refreshToken({\n *         refreshToken: storedRefreshToken,\n *       });\n *\n *       // Store new tokens\n *       await SecureStore.setItemAsync('accessToken', accessToken);\n *       await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *     } catch (error) {\n *       // Refresh token is invalid or expired\n *       navigation.navigate('Login');\n *     }\n *   };\n *\n *   return { handleTokenExpired };\n * }\n * ```\n *\n * @example\n * Integrate with SDK configuration:\n * ```tsx\n * import { configureSDK } from '@growsober/sdk';\n *\n * configureSDK({\n *   baseURL: 'https://api.growsober.app',\n *   getAccessToken: async () => {\n *     return await SecureStore.getItemAsync('accessToken');\n *   },\n *   refreshAccessToken: async () => {\n *     const refreshToken = await SecureStore.getItemAsync('refreshToken');\n *     const { accessToken, refreshToken: newRefreshToken } = await fetch('/auth/refresh', {\n *       method: 'POST',\n *       body: JSON.stringify({ refreshToken }),\n *     }).then(r => r.json());\n *\n *     await SecureStore.setItemAsync('accessToken', accessToken);\n *     await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *\n *     return accessToken;\n *   },\n *   onUnauthorized: () => {\n *     navigation.navigate('Login');\n *   },\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRefreshAuthToken(\n  options?: Omit<UseMutationOptions<TokenResponse, Error, RefreshTokenRequest>, 'mutationFn'>\n): UseMutationResult<TokenResponse, Error, RefreshTokenRequest> {\n  return useMutation({\n    mutationFn: async (data: RefreshTokenRequest): Promise<TokenResponse> => {\n      const client = getApiClient();\n      const response = await client.post<TokenResponse>('/auth/refresh', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Authenticate with Firebase ID token\n *\n * @description\n * Authenticates a user using a Firebase ID token.\n * Creates a new user account if one doesn't exist, or logs in an existing user.\n * Returns GrowSober authentication tokens and user information.\n *\n * @endpoint POST /auth/firebase\n *\n * @example\n * ```tsx\n * import { useFirebaseAuth } from '@growsober/sdk';\n * import { signInWithPhoneNumber } from 'firebase/auth';\n *\n * function PhoneAuthScreen() {\n *   const { mutate: firebaseAuth, isPending } = useFirebaseAuth({\n *     onSuccess: (data) => {\n *       // Store GrowSober tokens\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.user.onboardingCompleted) {\n *         navigation.navigate('Home');\n *       } else {\n *         navigation.navigate('Onboarding');\n *       }\n *     },\n *     onError: (error) => {\n *       Alert.alert('Authentication failed', error.message);\n *     },\n *   });\n *\n *   const handlePhoneAuth = async (phoneNumber: string) => {\n *     try {\n *       // Firebase authentication flow\n *       const confirmation = await signInWithPhoneNumber(auth, phoneNumber);\n *       const code = await promptUserForCode(); // Your UI to get verification code\n *       const credential = await confirmation.confirm(code);\n *\n *       // Get Firebase ID token\n *       const idToken = await credential.user.getIdToken();\n *\n *       // Authenticate with GrowSober backend\n *       firebaseAuth({ idToken });\n *     } catch (error) {\n *       console.error('Phone auth error:', error);\n *     }\n *   };\n *\n *   return <PhoneInput onSubmit={handlePhoneAuth} disabled={isPending} />;\n * }\n * ```\n *\n * @example\n * With Google Sign-In:\n * ```tsx\n * import { GoogleSignin } from '@react-native-google-signin/google-signin';\n *\n * const handleGoogleSignIn = async () => {\n *   const { idToken } = await GoogleSignin.signIn();\n *   firebaseAuth({ idToken });\n * };\n * ```\n *\n * @example\n * With Apple Sign-In:\n * ```tsx\n * import * as AppleAuthentication from 'expo-apple-authentication';\n *\n * const handleAppleSignIn = async () => {\n *   const credential = await AppleAuthentication.signInAsync({\n *     requestedScopes: [\n *       AppleAuthentication.AppleAuthenticationScope.FULL_NAME,\n *       AppleAuthentication.AppleAuthenticationScope.EMAIL,\n *     ],\n *   });\n *   firebaseAuth({ idToken: credential.identityToken });\n * };\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useFirebaseAuth(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, FirebaseAuthRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, FirebaseAuthRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: FirebaseAuthRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/firebase', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n"]}
334
+ /**
335
+ * Send OTP to phone number
336
+ *
337
+ * @description
338
+ * Sends a verification code to the specified phone number via SMS or voice call.
339
+ * Used for passwordless authentication.
340
+ *
341
+ * @endpoint POST /auth/phone/send-otp
342
+ *
343
+ * @example
344
+ * ```tsx
345
+ * import { useSendOtp } from '@growsober/sdk';
346
+ *
347
+ * function PhoneInputScreen() {
348
+ * const { mutate: sendOtp, isPending, error } = useSendOtp({
349
+ * onSuccess: (data) => {
350
+ * console.log('OTP sent to:', data.phone);
351
+ * navigation.navigate('VerifyOtp', { phone });
352
+ * },
353
+ * onError: (error) => {
354
+ * Alert.alert('Error', error.message);
355
+ * },
356
+ * });
357
+ *
358
+ * const handleSend = () => {
359
+ * sendOtp({ phone: '+1234567890', channel: 'sms' });
360
+ * };
361
+ *
362
+ * return <Button onPress={handleSend} disabled={isPending} />;
363
+ * }
364
+ * ```
365
+ *
366
+ * @param options - TanStack Query mutation options
367
+ * @returns TanStack Query mutation result
368
+ */
369
+ function useSendOtp(options) {
370
+ return (0, react_query_1.useMutation)({
371
+ mutationFn: async (data) => {
372
+ const client = (0, client_1.getApiClient)();
373
+ const response = await client.post('/auth/phone/send-otp', data);
374
+ return response.data;
375
+ },
376
+ ...options,
377
+ });
378
+ }
379
+ /**
380
+ * Verify OTP and authenticate
381
+ *
382
+ * @description
383
+ * Verifies the OTP code sent to the phone number and authenticates the user.
384
+ * If the user doesn't exist, a new account is created automatically.
385
+ * Returns authentication tokens and user information.
386
+ *
387
+ * @endpoint POST /auth/phone/verify-otp
388
+ *
389
+ * @example
390
+ * ```tsx
391
+ * import { useVerifyOtp } from '@growsober/sdk';
392
+ *
393
+ * function VerifyOtpScreen({ phone }) {
394
+ * const { mutate: verifyOtp, isPending, error } = useVerifyOtp({
395
+ * onSuccess: async (data) => {
396
+ * await SecureStore.setItemAsync('accessToken', data.accessToken);
397
+ * await SecureStore.setItemAsync('refreshToken', data.refreshToken);
398
+ *
399
+ * if (data.isNewUser) {
400
+ * navigation.navigate('Onboarding');
401
+ * } else {
402
+ * navigation.navigate('Home');
403
+ * }
404
+ * },
405
+ * });
406
+ *
407
+ * const handleVerify = (code: string) => {
408
+ * verifyOtp({ phone, code });
409
+ * };
410
+ *
411
+ * return <OtpInput onComplete={handleVerify} disabled={isPending} />;
412
+ * }
413
+ * ```
414
+ *
415
+ * @param options - TanStack Query mutation options
416
+ * @returns TanStack Query mutation result
417
+ */
418
+ function useVerifyOtp(options) {
419
+ const queryClient = (0, react_query_1.useQueryClient)();
420
+ return (0, react_query_1.useMutation)({
421
+ mutationFn: async (data) => {
422
+ const client = (0, client_1.getApiClient)();
423
+ const response = await client.post('/auth/phone/verify-otp', data);
424
+ return response.data;
425
+ },
426
+ onSuccess: (data, variables, context) => {
427
+ // Invalidate current user query to trigger refetch with new token
428
+ queryClient.invalidateQueries({ queryKey: users_1.userKeys.me() });
429
+ },
430
+ ...options,
431
+ });
432
+ }
433
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/api/mutations/auth.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Auth Mutation Hooks\n *\n * TanStack Query mutation hooks for authentication-related write operations.\n * These hooks handle user registration, login, token refresh, and Firebase authentication.\n *\n * @module api/mutations/auth\n */\n\nimport { useMutation, UseMutationOptions, UseMutationResult, useQueryClient } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\nimport { userKeys } from '../queries/users';\nimport type {\n  RegisterRequest,\n  LoginRequest,\n  RefreshTokenRequest,\n  FirebaseAuthRequest,\n  AuthResponse,\n  TokenResponse,\n  SendOtpRequest,\n  OtpSentResponse,\n  VerifyOtpRequest,\n} from '../types';\n\n// ============================================================================\n// MUTATION HOOKS\n// ============================================================================\n\n/**\n * Register a new user account\n *\n * @description\n * Creates a new user account with email/phone and password.\n * Returns authentication tokens and user information upon successful registration.\n *\n * @endpoint POST /auth/register\n *\n * @example\n * ```tsx\n * import { useRegister } from '@growsober/sdk';\n *\n * function RegisterForm() {\n *   const { mutate: register, isPending, error } = useRegister({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Onboarding');\n *     },\n *     onError: (error) => {\n *       Alert.alert('Registration failed', error.message);\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     register({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *       name: 'John Doe',\n *     });\n *   };\n *\n *   return <Button onPress={handleSubmit} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRegister(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, RegisterRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, RegisterRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: RegisterRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/register', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Login with email/phone and password\n *\n * @description\n * Authenticates an existing user with their credentials.\n * Returns authentication tokens and user information upon successful login.\n *\n * @endpoint POST /auth/login\n *\n * @example\n * ```tsx\n * import { useLogin } from '@growsober/sdk';\n *\n * function LoginForm() {\n *   const { mutate: login, isPending, error } = useLogin({\n *     onSuccess: (data) => {\n *       // Store tokens securely\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *       navigation.navigate('Home');\n *     },\n *   });\n *\n *   const handleSubmit = () => {\n *     login({\n *       email: 'user@example.com',\n *       password: 'SecurePassword123!',\n *     });\n *   };\n *\n *   return (\n *     <form onSubmit={handleSubmit}>\n *       <input type=\"email\" name=\"email\" />\n *       <input type=\"password\" name=\"password\" />\n *       <button type=\"submit\" disabled={isPending}>\n *         {isPending ? 'Logging in...' : 'Login'}\n *       </button>\n *       {error && <p className=\"error\">{error.message}</p>}\n *     </form>\n *   );\n * }\n * ```\n *\n * @example\n * Login with phone number:\n * ```tsx\n * login({\n *   phone: '+1234567890',\n *   password: 'SecurePassword123!',\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useLogin(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, LoginRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, LoginRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: LoginRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/login', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Refresh access token using refresh token\n *\n * @description\n * Obtains a new access token using a valid refresh token.\n * Should be called when the access token expires.\n *\n * @endpoint POST /auth/refresh\n *\n * @example\n * ```tsx\n * import { useRefreshAuthToken } from '@growsober/sdk';\n *\n * function useTokenRefresh() {\n *   const { mutateAsync: refreshToken } = useRefreshAuthToken();\n *\n *   const handleTokenExpired = async () => {\n *     const storedRefreshToken = await SecureStore.getItemAsync('refreshToken');\n *\n *     if (!storedRefreshToken) {\n *       navigation.navigate('Login');\n *       return;\n *     }\n *\n *     try {\n *       const { accessToken, refreshToken: newRefreshToken } = await refreshToken({\n *         refreshToken: storedRefreshToken,\n *       });\n *\n *       // Store new tokens\n *       await SecureStore.setItemAsync('accessToken', accessToken);\n *       await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *     } catch (error) {\n *       // Refresh token is invalid or expired\n *       navigation.navigate('Login');\n *     }\n *   };\n *\n *   return { handleTokenExpired };\n * }\n * ```\n *\n * @example\n * Integrate with SDK configuration:\n * ```tsx\n * import { configureSDK } from '@growsober/sdk';\n *\n * configureSDK({\n *   baseURL: 'https://api.growsober.app',\n *   getAccessToken: async () => {\n *     return await SecureStore.getItemAsync('accessToken');\n *   },\n *   refreshAccessToken: async () => {\n *     const refreshToken = await SecureStore.getItemAsync('refreshToken');\n *     const { accessToken, refreshToken: newRefreshToken } = await fetch('/auth/refresh', {\n *       method: 'POST',\n *       body: JSON.stringify({ refreshToken }),\n *     }).then(r => r.json());\n *\n *     await SecureStore.setItemAsync('accessToken', accessToken);\n *     await SecureStore.setItemAsync('refreshToken', newRefreshToken);\n *\n *     return accessToken;\n *   },\n *   onUnauthorized: () => {\n *     navigation.navigate('Login');\n *   },\n * });\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useRefreshAuthToken(\n  options?: Omit<UseMutationOptions<TokenResponse, Error, RefreshTokenRequest>, 'mutationFn'>\n): UseMutationResult<TokenResponse, Error, RefreshTokenRequest> {\n  return useMutation({\n    mutationFn: async (data: RefreshTokenRequest): Promise<TokenResponse> => {\n      const client = getApiClient();\n      const response = await client.post<TokenResponse>('/auth/refresh', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Authenticate with Firebase ID token\n *\n * @description\n * Authenticates a user using a Firebase ID token.\n * Creates a new user account if one doesn't exist, or logs in an existing user.\n * Returns GrowSober authentication tokens and user information.\n *\n * @endpoint POST /auth/firebase\n *\n * @example\n * ```tsx\n * import { useFirebaseAuth } from '@growsober/sdk';\n * import { signInWithPhoneNumber } from 'firebase/auth';\n *\n * function PhoneAuthScreen() {\n *   const { mutate: firebaseAuth, isPending } = useFirebaseAuth({\n *     onSuccess: (data) => {\n *       // Store GrowSober tokens\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.user.onboardingCompleted) {\n *         navigation.navigate('Home');\n *       } else {\n *         navigation.navigate('Onboarding');\n *       }\n *     },\n *     onError: (error) => {\n *       Alert.alert('Authentication failed', error.message);\n *     },\n *   });\n *\n *   const handlePhoneAuth = async (phoneNumber: string) => {\n *     try {\n *       // Firebase authentication flow\n *       const confirmation = await signInWithPhoneNumber(auth, phoneNumber);\n *       const code = await promptUserForCode(); // Your UI to get verification code\n *       const credential = await confirmation.confirm(code);\n *\n *       // Get Firebase ID token\n *       const idToken = await credential.user.getIdToken();\n *\n *       // Authenticate with GrowSober backend\n *       firebaseAuth({ idToken });\n *     } catch (error) {\n *       console.error('Phone auth error:', error);\n *     }\n *   };\n *\n *   return <PhoneInput onSubmit={handlePhoneAuth} disabled={isPending} />;\n * }\n * ```\n *\n * @example\n * With Google Sign-In:\n * ```tsx\n * import { GoogleSignin } from '@react-native-google-signin/google-signin';\n *\n * const handleGoogleSignIn = async () => {\n *   const { idToken } = await GoogleSignin.signIn();\n *   firebaseAuth({ idToken });\n * };\n * ```\n *\n * @example\n * With Apple Sign-In:\n * ```tsx\n * import * as AppleAuthentication from 'expo-apple-authentication';\n *\n * const handleAppleSignIn = async () => {\n *   const credential = await AppleAuthentication.signInAsync({\n *     requestedScopes: [\n *       AppleAuthentication.AppleAuthenticationScope.FULL_NAME,\n *       AppleAuthentication.AppleAuthenticationScope.EMAIL,\n *     ],\n *   });\n *   firebaseAuth({ idToken: credential.identityToken });\n * };\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useFirebaseAuth(\n  options?: Omit<UseMutationOptions<AuthResponse, Error, FirebaseAuthRequest>, 'mutationFn'>\n): UseMutationResult<AuthResponse, Error, FirebaseAuthRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: FirebaseAuthRequest): Promise<AuthResponse> => {\n      const client = getApiClient();\n      const response = await client.post<AuthResponse>('/auth/firebase', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n      // User's onSuccess is handled by spreading options\n    },\n    ...options,\n  });\n}\n\n/**\n * Response type for verify OTP including isNewUser flag\n */\nexport type VerifyOtpResponse = AuthResponse & { isNewUser: boolean };\n\n/**\n * Send OTP to phone number\n *\n * @description\n * Sends a verification code to the specified phone number via SMS or voice call.\n * Used for passwordless authentication.\n *\n * @endpoint POST /auth/phone/send-otp\n *\n * @example\n * ```tsx\n * import { useSendOtp } from '@growsober/sdk';\n *\n * function PhoneInputScreen() {\n *   const { mutate: sendOtp, isPending, error } = useSendOtp({\n *     onSuccess: (data) => {\n *       console.log('OTP sent to:', data.phone);\n *       navigation.navigate('VerifyOtp', { phone });\n *     },\n *     onError: (error) => {\n *       Alert.alert('Error', error.message);\n *     },\n *   });\n *\n *   const handleSend = () => {\n *     sendOtp({ phone: '+1234567890', channel: 'sms' });\n *   };\n *\n *   return <Button onPress={handleSend} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useSendOtp(\n  options?: Omit<UseMutationOptions<OtpSentResponse, Error, SendOtpRequest>, 'mutationFn'>\n): UseMutationResult<OtpSentResponse, Error, SendOtpRequest> {\n  return useMutation({\n    mutationFn: async (data: SendOtpRequest): Promise<OtpSentResponse> => {\n      const client = getApiClient();\n      const response = await client.post<OtpSentResponse>('/auth/phone/send-otp', data);\n      return response.data;\n    },\n    ...options,\n  });\n}\n\n/**\n * Verify OTP and authenticate\n *\n * @description\n * Verifies the OTP code sent to the phone number and authenticates the user.\n * If the user doesn't exist, a new account is created automatically.\n * Returns authentication tokens and user information.\n *\n * @endpoint POST /auth/phone/verify-otp\n *\n * @example\n * ```tsx\n * import { useVerifyOtp } from '@growsober/sdk';\n *\n * function VerifyOtpScreen({ phone }) {\n *   const { mutate: verifyOtp, isPending, error } = useVerifyOtp({\n *     onSuccess: async (data) => {\n *       await SecureStore.setItemAsync('accessToken', data.accessToken);\n *       await SecureStore.setItemAsync('refreshToken', data.refreshToken);\n *\n *       if (data.isNewUser) {\n *         navigation.navigate('Onboarding');\n *       } else {\n *         navigation.navigate('Home');\n *       }\n *     },\n *   });\n *\n *   const handleVerify = (code: string) => {\n *     verifyOtp({ phone, code });\n *   };\n *\n *   return <OtpInput onComplete={handleVerify} disabled={isPending} />;\n * }\n * ```\n *\n * @param options - TanStack Query mutation options\n * @returns TanStack Query mutation result\n */\nexport function useVerifyOtp(\n  options?: Omit<UseMutationOptions<VerifyOtpResponse, Error, VerifyOtpRequest>, 'mutationFn'>\n): UseMutationResult<VerifyOtpResponse, Error, VerifyOtpRequest> {\n  const queryClient = useQueryClient();\n\n  return useMutation({\n    mutationFn: async (data: VerifyOtpRequest): Promise<VerifyOtpResponse> => {\n      const client = getApiClient();\n      const response = await client.post<VerifyOtpResponse>('/auth/phone/verify-otp', data);\n      return response.data;\n    },\n    onSuccess: (data, variables, context) => {\n      // Invalidate current user query to trigger refetch with new token\n      queryClient.invalidateQueries({ queryKey: userKeys.me() });\n    },\n    ...options,\n  });\n}\n"]}
@@ -0,0 +1,52 @@
1
+ /**
2
+ * Cities Query Hooks
3
+ *
4
+ * TanStack Query hooks for fetching city data.
5
+ * Cities are used for location-based features and user onboarding.
6
+ *
7
+ * @module api/queries/cities
8
+ */
9
+ import { UseQueryOptions, UseQueryResult } from '@tanstack/react-query';
10
+ export declare const cityKeys: {
11
+ all: readonly ["cities"];
12
+ lists: () => readonly ["cities", "list"];
13
+ list: (filters?: CityFilters) => readonly ["cities", "list", CityFilters | undefined];
14
+ details: () => readonly ["cities", "detail"];
15
+ detail: (id: string) => readonly ["cities", "detail", string];
16
+ featured: () => readonly ["cities", "featured"];
17
+ };
18
+ export interface CityResponse {
19
+ id: string;
20
+ name: string;
21
+ country: string;
22
+ countryCode: string | null;
23
+ timezone: string | null;
24
+ latitude: number | null;
25
+ longitude: number | null;
26
+ memberCount: number;
27
+ isFeatured: boolean;
28
+ region: string | null;
29
+ }
30
+ export interface CityFilters {
31
+ search?: string;
32
+ countryCode?: string;
33
+ isFeatured?: boolean;
34
+ }
35
+ export declare function useCities(filters?: CityFilters, options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>): UseQueryResult<CityResponse[]>;
36
+ /**
37
+ * Get featured cities
38
+ *
39
+ * @description
40
+ * Fetches only cities marked as featured.
41
+ * Useful for showing popular/recommended cities.
42
+ *
43
+ * @example
44
+ * ```tsx
45
+ * const { data: featuredCities } = useFeaturedCities();
46
+ * ```
47
+ *
48
+ * @param options - TanStack Query options
49
+ * @returns TanStack Query result with array of featured cities
50
+ */
51
+ export declare function useFeaturedCities(options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>): UseQueryResult<CityResponse[]>;
52
+ export declare function useCity(id: string, options?: Omit<UseQueryOptions<CityResponse>, 'queryKey' | 'queryFn'>): UseQueryResult<CityResponse>;
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ /**
3
+ * Cities Query Hooks
4
+ *
5
+ * TanStack Query hooks for fetching city data.
6
+ * Cities are used for location-based features and user onboarding.
7
+ *
8
+ * @module api/queries/cities
9
+ */
10
+ Object.defineProperty(exports, "__esModule", { value: true });
11
+ exports.cityKeys = void 0;
12
+ exports.useCities = useCities;
13
+ exports.useFeaturedCities = useFeaturedCities;
14
+ exports.useCity = useCity;
15
+ const react_query_1 = require("@tanstack/react-query");
16
+ const client_1 = require("../client");
17
+ // ============================================================================
18
+ // QUERY KEY FACTORY
19
+ // ============================================================================
20
+ exports.cityKeys = {
21
+ all: ['cities'],
22
+ lists: () => [...exports.cityKeys.all, 'list'],
23
+ list: (filters) => [...exports.cityKeys.lists(), filters],
24
+ details: () => [...exports.cityKeys.all, 'detail'],
25
+ detail: (id) => [...exports.cityKeys.details(), id],
26
+ featured: () => [...exports.cityKeys.all, 'featured'],
27
+ };
28
+ function useCities(filters, options) {
29
+ return (0, react_query_1.useQuery)({
30
+ queryKey: exports.cityKeys.list(filters),
31
+ queryFn: async () => {
32
+ const client = (0, client_1.getApiClient)();
33
+ const params = new URLSearchParams();
34
+ if (filters?.search)
35
+ params.append('search', filters.search);
36
+ if (filters?.countryCode)
37
+ params.append('countryCode', filters.countryCode);
38
+ if (filters?.isFeatured !== undefined)
39
+ params.append('isFeatured', String(filters.isFeatured));
40
+ const queryString = params.toString();
41
+ const url = `/api/v1/cities${queryString ? `?${queryString}` : ''}`;
42
+ const response = await client.get(url);
43
+ // Handle both wrapped and unwrapped response formats
44
+ return Array.isArray(response.data) ? response.data : response.data.data;
45
+ },
46
+ staleTime: 5 * 60 * 1000, // Cities don't change often, cache for 5 minutes
47
+ ...options,
48
+ });
49
+ }
50
+ /**
51
+ * Get featured cities
52
+ *
53
+ * @description
54
+ * Fetches only cities marked as featured.
55
+ * Useful for showing popular/recommended cities.
56
+ *
57
+ * @example
58
+ * ```tsx
59
+ * const { data: featuredCities } = useFeaturedCities();
60
+ * ```
61
+ *
62
+ * @param options - TanStack Query options
63
+ * @returns TanStack Query result with array of featured cities
64
+ */
65
+ function useFeaturedCities(options) {
66
+ return useCities({ isFeatured: true }, options);
67
+ }
68
+ function useCity(id, options) {
69
+ return (0, react_query_1.useQuery)({
70
+ queryKey: exports.cityKeys.detail(id),
71
+ queryFn: async () => {
72
+ const client = (0, client_1.getApiClient)();
73
+ const response = await client.get(`/api/v1/cities/${id}`);
74
+ // Handle both wrapped and unwrapped response formats
75
+ return response.data.data || response.data;
76
+ },
77
+ enabled: !!id,
78
+ staleTime: 5 * 60 * 1000,
79
+ ...options,
80
+ });
81
+ }
82
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cities.js","sourceRoot":"","sources":["../../../src/api/queries/cities.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Cities Query Hooks\n *\n * TanStack Query hooks for fetching city data.\n * Cities are used for location-based features and user onboarding.\n *\n * @module api/queries/cities\n */\n\nimport { useQuery, UseQueryOptions, UseQueryResult } from '@tanstack/react-query';\nimport { getApiClient } from '../client';\n\n// ============================================================================\n// QUERY KEY FACTORY\n// ============================================================================\n\nexport const cityKeys = {\n  all: ['cities'] as const,\n  lists: () => [...cityKeys.all, 'list'] as const,\n  list: (filters?: CityFilters) => [...cityKeys.lists(), filters] as const,\n  details: () => [...cityKeys.all, 'detail'] as const,\n  detail: (id: string) => [...cityKeys.details(), id] as const,\n  featured: () => [...cityKeys.all, 'featured'] as const,\n};\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport interface CityResponse {\n  id: string;\n  name: string;\n  country: string;\n  countryCode: string | null;\n  timezone: string | null;\n  latitude: number | null;\n  longitude: number | null;\n  memberCount: number;\n  isFeatured: boolean;\n  region: string | null;\n}\n\nexport interface CityFilters {\n  search?: string;\n  countryCode?: string;\n  isFeatured?: boolean;\n}\n\n// ============================================================================\n// QUERY HOOKS\n// ============================================================================\n\n/**\n * Get list of cities with optional filters\n *\n * @description\n * Fetches available cities for location selection.\n * Can be filtered by name search, country code, or featured status.\n * Results are ordered by featured status, member count, and name.\n *\n * @endpoint GET /api/v1/cities\n *\n * @example\n * ```tsx\n * import { useCities } from '@growsober/sdk';\n *\n * function CityPicker() {\n *   const { data: cities, isLoading } = useCities();\n *\n *   if (isLoading) return <Spinner />;\n *\n *   return (\n *     <select>\n *       {cities?.map(city => (\n *         <option key={city.id} value={city.id}>\n *           {city.name}, {city.country}\n *         </option>\n *       ))}\n *     </select>\n *   );\n * }\n * ```\n *\n * @example\n * With search filter:\n * ```tsx\n * const { data: cities } = useCities({ search: 'London' });\n * ```\n *\n * @example\n * Filter by country:\n * ```tsx\n * const { data: ukCities } = useCities({ countryCode: 'GB' });\n * ```\n *\n * @param filters - Optional filters for city search\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of cities\n */\ninterface CitiesApiResponse {\n  data: CityResponse[];\n  meta: {\n    timestamp: string;\n  };\n}\n\nexport function useCities(\n  filters?: CityFilters,\n  options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<CityResponse[]> {\n  return useQuery({\n    queryKey: cityKeys.list(filters),\n    queryFn: async (): Promise<CityResponse[]> => {\n      const client = getApiClient();\n      const params = new URLSearchParams();\n\n      if (filters?.search) params.append('search', filters.search);\n      if (filters?.countryCode) params.append('countryCode', filters.countryCode);\n      if (filters?.isFeatured !== undefined) params.append('isFeatured', String(filters.isFeatured));\n\n      const queryString = params.toString();\n      const url = `/api/v1/cities${queryString ? `?${queryString}` : ''}`;\n\n      const response = await client.get<CitiesApiResponse>(url);\n      // Handle both wrapped and unwrapped response formats\n      return Array.isArray(response.data) ? response.data : response.data.data;\n    },\n    staleTime: 5 * 60 * 1000, // Cities don't change often, cache for 5 minutes\n    ...options,\n  });\n}\n\n/**\n * Get featured cities\n *\n * @description\n * Fetches only cities marked as featured.\n * Useful for showing popular/recommended cities.\n *\n * @example\n * ```tsx\n * const { data: featuredCities } = useFeaturedCities();\n * ```\n *\n * @param options - TanStack Query options\n * @returns TanStack Query result with array of featured cities\n */\nexport function useFeaturedCities(\n  options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>\n): UseQueryResult<CityResponse[]> {\n  return useCities({ isFeatured: true }, options);\n}\n\n/**\n * Get a single city by ID\n *\n * @description\n * Fetches details for a specific city.\n *\n * @endpoint GET /api/v1/cities/{id}\n *\n * @example\n * ```tsx\n * const { data: city } = useCity('city-uuid');\n * ```\n *\n * @param id - City ID\n * @param options - TanStack Query options\n * @returns TanStack Query result with city data\n */\ninterface CityApiResponse {\n  data: CityResponse;\n  meta: {\n    timestamp: string;\n  };\n}\n\nexport function useCity(\n  id: string,\n  options?: Omit<UseQueryOptions<CityResponse>, 'queryKey' | 'queryFn'>\n): UseQueryResult<CityResponse> {\n  return useQuery({\n    queryKey: cityKeys.detail(id),\n    queryFn: async (): Promise<CityResponse> => {\n      const client = getApiClient();\n      const response = await client.get<CityApiResponse>(`/api/v1/cities/${id}`);\n      // Handle both wrapped and unwrapped response formats\n      return (response.data as any).data || response.data;\n    },\n    enabled: !!id,\n    staleTime: 5 * 60 * 1000,\n    ...options,\n  });\n}\n"]}
@@ -7,6 +7,7 @@ export * from './admin';
7
7
  export * from './auth';
8
8
  export * from './bookings';
9
9
  export * from './businesses';
10
+ export * from './cities';
10
11
  export * from './events';
11
12
  export * from './hubs';
12
13
  export * from './library';
@@ -23,6 +23,7 @@ __exportStar(require("./admin"), exports);
23
23
  __exportStar(require("./auth"), exports);
24
24
  __exportStar(require("./bookings"), exports);
25
25
  __exportStar(require("./businesses"), exports);
26
+ __exportStar(require("./cities"), exports);
26
27
  __exportStar(require("./events"), exports);
27
28
  __exportStar(require("./hubs"), exports);
28
29
  __exportStar(require("./library"), exports);
@@ -37,4 +38,4 @@ __exportStar(require("./ambassadors"), exports);
37
38
  __exportStar(require("./grow90"), exports);
38
39
  __exportStar(require("./matching"), exports);
39
40
  __exportStar(require("./event-chat"), exports);
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsMkNBQXlCO0FBQ3pCLHlDQUF1QjtBQUN2Qiw0Q0FBMEI7QUFDMUIsd0NBQXNCO0FBQ3RCLGtEQUFnQztBQUNoQywyQ0FBeUI7QUFDekIsa0RBQWdDO0FBQ2hDLDRDQUEwQjtBQUMxQiwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLGdEQUE4QjtBQUM5QiwyQ0FBeUI7QUFDekIsNkNBQTJCO0FBQzNCLCtDQUE2QiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUXVlcnkgSG9va3NcbiAqXG4gKiBSZS1leHBvcnRzIGFsbCBxdWVyeSBob29rcyBmb3IgQVBJIGVuZHBvaW50cy5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2FkbWluJztcbmV4cG9ydCAqIGZyb20gJy4vYXV0aCc7XG5leHBvcnQgKiBmcm9tICcuL2Jvb2tpbmdzJztcbmV4cG9ydCAqIGZyb20gJy4vYnVzaW5lc3Nlcyc7XG5leHBvcnQgKiBmcm9tICcuL2V2ZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2h1YnMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWJyYXJ5JztcbmV4cG9ydCAqIGZyb20gJy4vbWFwJztcbmV4cG9ydCAqIGZyb20gJy4vbm90aWZpY2F0aW9ucyc7XG5leHBvcnQgKiBmcm9tICcuL29mZmVycyc7XG5leHBvcnQgKiBmcm9tICcuL3N1YnNjcmlwdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdXBwb3J0JztcbmV4cG9ydCAqIGZyb20gJy4vdXNlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9qYWNrJztcbmV4cG9ydCAqIGZyb20gJy4vYW1iYXNzYWRvcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9ncm93OTAnO1xuZXhwb3J0ICogZnJvbSAnLi9tYXRjaGluZyc7XG5leHBvcnQgKiBmcm9tICcuL2V2ZW50LWNoYXQnO1xuIl19
41
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsMkNBQXlCO0FBQ3pCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFF1ZXJ5IEhvb2tzXG4gKlxuICogUmUtZXhwb3J0cyBhbGwgcXVlcnkgaG9va3MgZm9yIEFQSSBlbmRwb2ludHMuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9hZG1pbic7XG5leHBvcnQgKiBmcm9tICcuL2F1dGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ib29raW5ncyc7XG5leHBvcnQgKiBmcm9tICcuL2J1c2luZXNzZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9jaXRpZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9odWJzJztcbmV4cG9ydCAqIGZyb20gJy4vbGlicmFyeSc7XG5leHBvcnQgKiBmcm9tICcuL21hcCc7XG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9vZmZlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdWJzY3JpcHRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vc3VwcG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL3VzZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vamFjayc7XG5leHBvcnQgKiBmcm9tICcuL2FtYmFzc2Fkb3JzJztcbmV4cG9ydCAqIGZyb20gJy4vZ3JvdzkwJztcbmV4cG9ydCAqIGZyb20gJy4vbWF0Y2hpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudC1jaGF0JztcbiJdfQ==
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@growsober/sdk",
3
- "version": "1.0.1",
3
+ "version": "1.0.2",
4
4
  "description": "Shared TypeScript SDK for GrowSober API - TanStack Query hooks, API client, and utilities",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.js",
@@ -17,6 +17,9 @@ import type {
17
17
  FirebaseAuthRequest,
18
18
  AuthResponse,
19
19
  TokenResponse,
20
+ SendOtpRequest,
21
+ OtpSentResponse,
22
+ VerifyOtpRequest,
20
23
  } from '../types';
21
24
 
22
25
  // ============================================================================
@@ -348,3 +351,114 @@ export function useFirebaseAuth(
348
351
  ...options,
349
352
  });
350
353
  }
354
+
355
+ /**
356
+ * Response type for verify OTP including isNewUser flag
357
+ */
358
+ export type VerifyOtpResponse = AuthResponse & { isNewUser: boolean };
359
+
360
+ /**
361
+ * Send OTP to phone number
362
+ *
363
+ * @description
364
+ * Sends a verification code to the specified phone number via SMS or voice call.
365
+ * Used for passwordless authentication.
366
+ *
367
+ * @endpoint POST /auth/phone/send-otp
368
+ *
369
+ * @example
370
+ * ```tsx
371
+ * import { useSendOtp } from '@growsober/sdk';
372
+ *
373
+ * function PhoneInputScreen() {
374
+ * const { mutate: sendOtp, isPending, error } = useSendOtp({
375
+ * onSuccess: (data) => {
376
+ * console.log('OTP sent to:', data.phone);
377
+ * navigation.navigate('VerifyOtp', { phone });
378
+ * },
379
+ * onError: (error) => {
380
+ * Alert.alert('Error', error.message);
381
+ * },
382
+ * });
383
+ *
384
+ * const handleSend = () => {
385
+ * sendOtp({ phone: '+1234567890', channel: 'sms' });
386
+ * };
387
+ *
388
+ * return <Button onPress={handleSend} disabled={isPending} />;
389
+ * }
390
+ * ```
391
+ *
392
+ * @param options - TanStack Query mutation options
393
+ * @returns TanStack Query mutation result
394
+ */
395
+ export function useSendOtp(
396
+ options?: Omit<UseMutationOptions<OtpSentResponse, Error, SendOtpRequest>, 'mutationFn'>
397
+ ): UseMutationResult<OtpSentResponse, Error, SendOtpRequest> {
398
+ return useMutation({
399
+ mutationFn: async (data: SendOtpRequest): Promise<OtpSentResponse> => {
400
+ const client = getApiClient();
401
+ const response = await client.post<OtpSentResponse>('/auth/phone/send-otp', data);
402
+ return response.data;
403
+ },
404
+ ...options,
405
+ });
406
+ }
407
+
408
+ /**
409
+ * Verify OTP and authenticate
410
+ *
411
+ * @description
412
+ * Verifies the OTP code sent to the phone number and authenticates the user.
413
+ * If the user doesn't exist, a new account is created automatically.
414
+ * Returns authentication tokens and user information.
415
+ *
416
+ * @endpoint POST /auth/phone/verify-otp
417
+ *
418
+ * @example
419
+ * ```tsx
420
+ * import { useVerifyOtp } from '@growsober/sdk';
421
+ *
422
+ * function VerifyOtpScreen({ phone }) {
423
+ * const { mutate: verifyOtp, isPending, error } = useVerifyOtp({
424
+ * onSuccess: async (data) => {
425
+ * await SecureStore.setItemAsync('accessToken', data.accessToken);
426
+ * await SecureStore.setItemAsync('refreshToken', data.refreshToken);
427
+ *
428
+ * if (data.isNewUser) {
429
+ * navigation.navigate('Onboarding');
430
+ * } else {
431
+ * navigation.navigate('Home');
432
+ * }
433
+ * },
434
+ * });
435
+ *
436
+ * const handleVerify = (code: string) => {
437
+ * verifyOtp({ phone, code });
438
+ * };
439
+ *
440
+ * return <OtpInput onComplete={handleVerify} disabled={isPending} />;
441
+ * }
442
+ * ```
443
+ *
444
+ * @param options - TanStack Query mutation options
445
+ * @returns TanStack Query mutation result
446
+ */
447
+ export function useVerifyOtp(
448
+ options?: Omit<UseMutationOptions<VerifyOtpResponse, Error, VerifyOtpRequest>, 'mutationFn'>
449
+ ): UseMutationResult<VerifyOtpResponse, Error, VerifyOtpRequest> {
450
+ const queryClient = useQueryClient();
451
+
452
+ return useMutation({
453
+ mutationFn: async (data: VerifyOtpRequest): Promise<VerifyOtpResponse> => {
454
+ const client = getApiClient();
455
+ const response = await client.post<VerifyOtpResponse>('/auth/phone/verify-otp', data);
456
+ return response.data;
457
+ },
458
+ onSuccess: (data, variables, context) => {
459
+ // Invalidate current user query to trigger refetch with new token
460
+ queryClient.invalidateQueries({ queryKey: userKeys.me() });
461
+ },
462
+ ...options,
463
+ });
464
+ }
@@ -0,0 +1,194 @@
1
+ /**
2
+ * Cities Query Hooks
3
+ *
4
+ * TanStack Query hooks for fetching city data.
5
+ * Cities are used for location-based features and user onboarding.
6
+ *
7
+ * @module api/queries/cities
8
+ */
9
+
10
+ import { useQuery, UseQueryOptions, UseQueryResult } from '@tanstack/react-query';
11
+ import { getApiClient } from '../client';
12
+
13
+ // ============================================================================
14
+ // QUERY KEY FACTORY
15
+ // ============================================================================
16
+
17
+ export const cityKeys = {
18
+ all: ['cities'] as const,
19
+ lists: () => [...cityKeys.all, 'list'] as const,
20
+ list: (filters?: CityFilters) => [...cityKeys.lists(), filters] as const,
21
+ details: () => [...cityKeys.all, 'detail'] as const,
22
+ detail: (id: string) => [...cityKeys.details(), id] as const,
23
+ featured: () => [...cityKeys.all, 'featured'] as const,
24
+ };
25
+
26
+ // ============================================================================
27
+ // TYPES
28
+ // ============================================================================
29
+
30
+ export interface CityResponse {
31
+ id: string;
32
+ name: string;
33
+ country: string;
34
+ countryCode: string | null;
35
+ timezone: string | null;
36
+ latitude: number | null;
37
+ longitude: number | null;
38
+ memberCount: number;
39
+ isFeatured: boolean;
40
+ region: string | null;
41
+ }
42
+
43
+ export interface CityFilters {
44
+ search?: string;
45
+ countryCode?: string;
46
+ isFeatured?: boolean;
47
+ }
48
+
49
+ // ============================================================================
50
+ // QUERY HOOKS
51
+ // ============================================================================
52
+
53
+ /**
54
+ * Get list of cities with optional filters
55
+ *
56
+ * @description
57
+ * Fetches available cities for location selection.
58
+ * Can be filtered by name search, country code, or featured status.
59
+ * Results are ordered by featured status, member count, and name.
60
+ *
61
+ * @endpoint GET /api/v1/cities
62
+ *
63
+ * @example
64
+ * ```tsx
65
+ * import { useCities } from '@growsober/sdk';
66
+ *
67
+ * function CityPicker() {
68
+ * const { data: cities, isLoading } = useCities();
69
+ *
70
+ * if (isLoading) return <Spinner />;
71
+ *
72
+ * return (
73
+ * <select>
74
+ * {cities?.map(city => (
75
+ * <option key={city.id} value={city.id}>
76
+ * {city.name}, {city.country}
77
+ * </option>
78
+ * ))}
79
+ * </select>
80
+ * );
81
+ * }
82
+ * ```
83
+ *
84
+ * @example
85
+ * With search filter:
86
+ * ```tsx
87
+ * const { data: cities } = useCities({ search: 'London' });
88
+ * ```
89
+ *
90
+ * @example
91
+ * Filter by country:
92
+ * ```tsx
93
+ * const { data: ukCities } = useCities({ countryCode: 'GB' });
94
+ * ```
95
+ *
96
+ * @param filters - Optional filters for city search
97
+ * @param options - TanStack Query options
98
+ * @returns TanStack Query result with array of cities
99
+ */
100
+ interface CitiesApiResponse {
101
+ data: CityResponse[];
102
+ meta: {
103
+ timestamp: string;
104
+ };
105
+ }
106
+
107
+ export function useCities(
108
+ filters?: CityFilters,
109
+ options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>
110
+ ): UseQueryResult<CityResponse[]> {
111
+ return useQuery({
112
+ queryKey: cityKeys.list(filters),
113
+ queryFn: async (): Promise<CityResponse[]> => {
114
+ const client = getApiClient();
115
+ const params = new URLSearchParams();
116
+
117
+ if (filters?.search) params.append('search', filters.search);
118
+ if (filters?.countryCode) params.append('countryCode', filters.countryCode);
119
+ if (filters?.isFeatured !== undefined) params.append('isFeatured', String(filters.isFeatured));
120
+
121
+ const queryString = params.toString();
122
+ const url = `/api/v1/cities${queryString ? `?${queryString}` : ''}`;
123
+
124
+ const response = await client.get<CitiesApiResponse>(url);
125
+ // Handle both wrapped and unwrapped response formats
126
+ return Array.isArray(response.data) ? response.data : response.data.data;
127
+ },
128
+ staleTime: 5 * 60 * 1000, // Cities don't change often, cache for 5 minutes
129
+ ...options,
130
+ });
131
+ }
132
+
133
+ /**
134
+ * Get featured cities
135
+ *
136
+ * @description
137
+ * Fetches only cities marked as featured.
138
+ * Useful for showing popular/recommended cities.
139
+ *
140
+ * @example
141
+ * ```tsx
142
+ * const { data: featuredCities } = useFeaturedCities();
143
+ * ```
144
+ *
145
+ * @param options - TanStack Query options
146
+ * @returns TanStack Query result with array of featured cities
147
+ */
148
+ export function useFeaturedCities(
149
+ options?: Omit<UseQueryOptions<CityResponse[]>, 'queryKey' | 'queryFn'>
150
+ ): UseQueryResult<CityResponse[]> {
151
+ return useCities({ isFeatured: true }, options);
152
+ }
153
+
154
+ /**
155
+ * Get a single city by ID
156
+ *
157
+ * @description
158
+ * Fetches details for a specific city.
159
+ *
160
+ * @endpoint GET /api/v1/cities/{id}
161
+ *
162
+ * @example
163
+ * ```tsx
164
+ * const { data: city } = useCity('city-uuid');
165
+ * ```
166
+ *
167
+ * @param id - City ID
168
+ * @param options - TanStack Query options
169
+ * @returns TanStack Query result with city data
170
+ */
171
+ interface CityApiResponse {
172
+ data: CityResponse;
173
+ meta: {
174
+ timestamp: string;
175
+ };
176
+ }
177
+
178
+ export function useCity(
179
+ id: string,
180
+ options?: Omit<UseQueryOptions<CityResponse>, 'queryKey' | 'queryFn'>
181
+ ): UseQueryResult<CityResponse> {
182
+ return useQuery({
183
+ queryKey: cityKeys.detail(id),
184
+ queryFn: async (): Promise<CityResponse> => {
185
+ const client = getApiClient();
186
+ const response = await client.get<CityApiResponse>(`/api/v1/cities/${id}`);
187
+ // Handle both wrapped and unwrapped response formats
188
+ return (response.data as any).data || response.data;
189
+ },
190
+ enabled: !!id,
191
+ staleTime: 5 * 60 * 1000,
192
+ ...options,
193
+ });
194
+ }
@@ -8,6 +8,7 @@ export * from './admin';
8
8
  export * from './auth';
9
9
  export * from './bookings';
10
10
  export * from './businesses';
11
+ export * from './cities';
11
12
  export * from './events';
12
13
  export * from './hubs';
13
14
  export * from './library';