@growsober/sdk 1.0.1 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/mutations/auth.d.ts +83 -1
- package/dist/api/mutations/auth.js +106 -5
- package/dist/api/queries/cities.d.ts +52 -0
- package/dist/api/queries/cities.js +82 -0
- package/dist/api/queries/index.d.ts +1 -0
- package/dist/api/queries/index.js +2 -1
- package/package.json +1 -1
- package/src/api/mutations/auth.ts +118 -4
- package/src/api/queries/cities.ts +194 -0
- package/src/api/queries/index.ts +1 -0
|
@@ -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");
|
|
@@ -64,7 +66,7 @@ function useRegister(options) {
|
|
|
64
66
|
return (0, react_query_1.useMutation)({
|
|
65
67
|
mutationFn: async (data) => {
|
|
66
68
|
const client = (0, client_1.getApiClient)();
|
|
67
|
-
const response = await client.post('/auth/register', data);
|
|
69
|
+
const response = await client.post('/api/v1/auth/register', data);
|
|
68
70
|
return response.data;
|
|
69
71
|
},
|
|
70
72
|
onSuccess: (data, variables, context) => {
|
|
@@ -135,7 +137,7 @@ function useLogin(options) {
|
|
|
135
137
|
return (0, react_query_1.useMutation)({
|
|
136
138
|
mutationFn: async (data) => {
|
|
137
139
|
const client = (0, client_1.getApiClient)();
|
|
138
|
-
const response = await client.post('/auth/login', data);
|
|
140
|
+
const response = await client.post('/api/v1/auth/login', data);
|
|
139
141
|
return response.data;
|
|
140
142
|
},
|
|
141
143
|
onSuccess: (data, variables, context) => {
|
|
@@ -223,7 +225,7 @@ function useRefreshAuthToken(options) {
|
|
|
223
225
|
return (0, react_query_1.useMutation)({
|
|
224
226
|
mutationFn: async (data) => {
|
|
225
227
|
const client = (0, client_1.getApiClient)();
|
|
226
|
-
const response = await client.post('/auth/refresh', data);
|
|
228
|
+
const response = await client.post('/api/v1/auth/refresh', data);
|
|
227
229
|
return response.data;
|
|
228
230
|
},
|
|
229
231
|
...options,
|
|
@@ -318,7 +320,7 @@ function useFirebaseAuth(options) {
|
|
|
318
320
|
return (0, react_query_1.useMutation)({
|
|
319
321
|
mutationFn: async (data) => {
|
|
320
322
|
const client = (0, client_1.getApiClient)();
|
|
321
|
-
const response = await client.post('/auth/firebase', data);
|
|
323
|
+
const response = await client.post('/api/v1/auth/firebase', data);
|
|
322
324
|
return response.data;
|
|
323
325
|
},
|
|
324
326
|
onSuccess: (data, variables, context) => {
|
|
@@ -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('/api/v1/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('/api/v1/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":";AAAA;;;;;;;GAOG;;AA8DH,kCAkBC;AAyDD,4BAkBC;AA2ED,kDAWC;AAsFD,0CAkBC;AA0CD,gCAWC;AAyCD,oCAiBC;AAtcD,uDAA2G;AAC3G,sCAAyC;AACzC,4CAA4C;AAa5C,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwCG;AACH,SAAgB,WAAW,CACzB,OAAsF;IAEtF,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAqB,EAAyB,EAAE;YACjE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAe,uBAAuB,EAAE,IAAI,CAAC,CAAC;YAChF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,kEAAkE;YAClE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3D,mDAAmD;QACrD,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDG;AACH,SAAgB,QAAQ,CACtB,OAAmF;IAEnF,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAkB,EAAyB,EAAE;YAC9D,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAe,oBAAoB,EAAE,IAAI,CAAC,CAAC;YAC7E,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,kEAAkE;YAClE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3D,mDAAmD;QACrD,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwEG;AACH,SAAgB,mBAAmB,CACjC,OAA2F;IAE3F,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAyB,EAA0B,EAAE;YACtE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAgB,sBAAsB,EAAE,IAAI,CAAC,CAAC;YAChF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmFG;AACH,SAAgB,eAAe,CAC7B,OAA0F;IAE1F,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAyB,EAAyB,EAAE;YACrE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAe,uBAAuB,EAAE,IAAI,CAAC,CAAC;YAChF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,kEAAkE;YAClE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3D,mDAAmD;QACrD,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAOD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkCG;AACH,SAAgB,UAAU,CACxB,OAAwF;IAExF,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAoB,EAA4B,EAAE;YACnE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAkB,6BAA6B,EAAE,IAAI,CAAC,CAAC;YACzF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCG;AACH,SAAgB,YAAY,CAC1B,OAA4F;IAE5F,MAAM,WAAW,GAAG,IAAA,4BAAc,GAAE,CAAC;IAErC,OAAO,IAAA,yBAAW,EAAC;QACjB,UAAU,EAAE,KAAK,EAAE,IAAsB,EAA8B,EAAE;YACvE,MAAM,MAAM,GAAG,IAAA,qBAAY,GAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,IAAI,CAAoB,+BAA+B,EAAE,IAAI,CAAC,CAAC;YAC7F,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,SAAS,EAAE,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE;YACtC,kEAAkE;YAClE,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,gBAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QAC7D,CAAC;QACD,GAAG,OAAO;KACX,CAAC,CAAC;AACL,CAAC","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>('/api/v1/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>('/api/v1/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>('/api/v1/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>('/api/v1/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>('/api/v1/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>('/api/v1/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"]}
|
|
@@ -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,
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvYXBpL3F1ZXJpZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwwQ0FBd0I7QUFDeEIseUNBQXVCO0FBQ3ZCLDZDQUEyQjtBQUMzQiwrQ0FBNkI7QUFDN0IsMkNBQXlCO0FBQ3pCLDJDQUF5QjtBQUN6Qix5Q0FBdUI7QUFDdkIsNENBQTBCO0FBQzFCLHdDQUFzQjtBQUN0QixrREFBZ0M7QUFDaEMsMkNBQXlCO0FBQ3pCLGtEQUFnQztBQUNoQyw0Q0FBMEI7QUFDMUIsMENBQXdCO0FBQ3hCLHlDQUF1QjtBQUN2QixnREFBOEI7QUFDOUIsMkNBQXlCO0FBQ3pCLDZDQUEyQjtBQUMzQiwrQ0FBNkIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFF1ZXJ5IEhvb2tzXG4gKlxuICogUmUtZXhwb3J0cyBhbGwgcXVlcnkgaG9va3MgZm9yIEFQSSBlbmRwb2ludHMuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9hZG1pbic7XG5leHBvcnQgKiBmcm9tICcuL2F1dGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ib29raW5ncyc7XG5leHBvcnQgKiBmcm9tICcuL2J1c2luZXNzZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9jaXRpZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9odWJzJztcbmV4cG9ydCAqIGZyb20gJy4vbGlicmFyeSc7XG5leHBvcnQgKiBmcm9tICcuL21hcCc7XG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9vZmZlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdWJzY3JpcHRpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vc3VwcG9ydCc7XG5leHBvcnQgKiBmcm9tICcuL3VzZXJzJztcbmV4cG9ydCAqIGZyb20gJy4vamFjayc7XG5leHBvcnQgKiBmcm9tICcuL2FtYmFzc2Fkb3JzJztcbmV4cG9ydCAqIGZyb20gJy4vZ3JvdzkwJztcbmV4cG9ydCAqIGZyb20gJy4vbWF0Y2hpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9ldmVudC1jaGF0JztcbiJdfQ==
|
package/package.json
CHANGED
|
@@ -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
|
// ============================================================================
|
|
@@ -72,7 +75,7 @@ export function useRegister(
|
|
|
72
75
|
return useMutation({
|
|
73
76
|
mutationFn: async (data: RegisterRequest): Promise<AuthResponse> => {
|
|
74
77
|
const client = getApiClient();
|
|
75
|
-
const response = await client.post<AuthResponse>('/auth/register', data);
|
|
78
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/register', data);
|
|
76
79
|
return response.data;
|
|
77
80
|
},
|
|
78
81
|
onSuccess: (data, variables, context) => {
|
|
@@ -147,7 +150,7 @@ export function useLogin(
|
|
|
147
150
|
return useMutation({
|
|
148
151
|
mutationFn: async (data: LoginRequest): Promise<AuthResponse> => {
|
|
149
152
|
const client = getApiClient();
|
|
150
|
-
const response = await client.post<AuthResponse>('/auth/login', data);
|
|
153
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/login', data);
|
|
151
154
|
return response.data;
|
|
152
155
|
},
|
|
153
156
|
onSuccess: (data, variables, context) => {
|
|
@@ -238,7 +241,7 @@ export function useRefreshAuthToken(
|
|
|
238
241
|
return useMutation({
|
|
239
242
|
mutationFn: async (data: RefreshTokenRequest): Promise<TokenResponse> => {
|
|
240
243
|
const client = getApiClient();
|
|
241
|
-
const response = await client.post<TokenResponse>('/auth/refresh', data);
|
|
244
|
+
const response = await client.post<TokenResponse>('/api/v1/auth/refresh', data);
|
|
242
245
|
return response.data;
|
|
243
246
|
},
|
|
244
247
|
...options,
|
|
@@ -337,7 +340,7 @@ export function useFirebaseAuth(
|
|
|
337
340
|
return useMutation({
|
|
338
341
|
mutationFn: async (data: FirebaseAuthRequest): Promise<AuthResponse> => {
|
|
339
342
|
const client = getApiClient();
|
|
340
|
-
const response = await client.post<AuthResponse>('/auth/firebase', data);
|
|
343
|
+
const response = await client.post<AuthResponse>('/api/v1/auth/firebase', data);
|
|
341
344
|
return response.data;
|
|
342
345
|
},
|
|
343
346
|
onSuccess: (data, variables, context) => {
|
|
@@ -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>('/api/v1/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>('/api/v1/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
|
+
}
|
package/src/api/queries/index.ts
CHANGED