@chem-po/firebase-native 0.0.16 → 0.0.18

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.
Files changed (212) hide show
  1. package/README.md +215 -0
  2. package/lib/commonjs/adapter/auth.js +431 -0
  3. package/lib/commonjs/adapter/auth.js.map +1 -0
  4. package/lib/commonjs/adapter/db.js +103 -0
  5. package/lib/commonjs/adapter/db.js.map +1 -0
  6. package/lib/commonjs/adapter/index.js +16 -0
  7. package/lib/commonjs/adapter/index.js.map +1 -0
  8. package/lib/commonjs/adapter/storage.js +52 -0
  9. package/lib/commonjs/adapter/storage.js.map +1 -0
  10. package/lib/commonjs/auth/functions.js +11 -0
  11. package/lib/commonjs/auth/functions.js.map +1 -0
  12. package/lib/commonjs/auth/index.js +17 -0
  13. package/lib/commonjs/auth/index.js.map +1 -0
  14. package/lib/commonjs/components/AuthenticatorVerify.js +90 -0
  15. package/lib/commonjs/components/AuthenticatorVerify.js.map +1 -0
  16. package/lib/commonjs/components/FirebaseSignIn.js +196 -0
  17. package/lib/commonjs/components/FirebaseSignIn.js.map +1 -0
  18. package/lib/commonjs/components/PhoneVerify.js +123 -0
  19. package/lib/commonjs/components/PhoneVerify.js.map +1 -0
  20. package/lib/commonjs/components/TwoFactorAuthModal.js +118 -0
  21. package/lib/commonjs/components/TwoFactorAuthModal.js.map +1 -0
  22. package/lib/commonjs/components/index.js +28 -0
  23. package/lib/commonjs/components/index.js.map +1 -0
  24. package/lib/commonjs/contexts/FirebaseContext.js +48 -0
  25. package/lib/commonjs/contexts/FirebaseContext.js.map +1 -0
  26. package/lib/commonjs/contexts/index.js +17 -0
  27. package/lib/commonjs/contexts/index.js.map +1 -0
  28. package/lib/commonjs/db/index.js +17 -0
  29. package/lib/commonjs/db/index.js.map +1 -0
  30. package/lib/commonjs/db/utils.js +120 -0
  31. package/lib/commonjs/db/utils.js.map +1 -0
  32. package/lib/commonjs/hooks/backend.js +12 -0
  33. package/lib/commonjs/hooks/backend.js.map +1 -0
  34. package/lib/commonjs/hooks/index.js +17 -0
  35. package/lib/commonjs/hooks/index.js.map +1 -0
  36. package/lib/commonjs/hooks/useAuthenticatorVerify.js +52 -0
  37. package/lib/commonjs/hooks/useAuthenticatorVerify.js.map +1 -0
  38. package/lib/commonjs/hooks/usePhoneVerify.js +83 -0
  39. package/lib/commonjs/hooks/usePhoneVerify.js.map +1 -0
  40. package/lib/commonjs/icons/Google.js +29 -0
  41. package/lib/commonjs/icons/Google.js.map +1 -0
  42. package/lib/commonjs/index.js +105 -0
  43. package/lib/commonjs/index.js.map +1 -0
  44. package/lib/commonjs/storage/index.js +17 -0
  45. package/lib/commonjs/storage/index.js.map +1 -0
  46. package/lib/commonjs/storage/utils.js +37 -0
  47. package/lib/commonjs/storage/utils.js.map +1 -0
  48. package/lib/commonjs/types/adapter.js +6 -0
  49. package/lib/commonjs/types/adapter.js.map +1 -0
  50. package/lib/commonjs/types/auth.js +6 -0
  51. package/lib/commonjs/types/auth.js.map +1 -0
  52. package/lib/commonjs/types/db.js +6 -0
  53. package/lib/commonjs/types/db.js.map +1 -0
  54. package/lib/commonjs/types/functions.js +6 -0
  55. package/lib/commonjs/types/functions.js.map +1 -0
  56. package/lib/commonjs/types/index.js +6 -0
  57. package/lib/commonjs/types/index.js.map +1 -0
  58. package/lib/commonjs/types/storage.js +6 -0
  59. package/lib/commonjs/types/storage.js.map +1 -0
  60. package/lib/commonjs/utils/validation.js +71 -0
  61. package/lib/commonjs/utils/validation.js.map +1 -0
  62. package/lib/module/adapter/auth.js +424 -0
  63. package/lib/module/adapter/auth.js.map +1 -0
  64. package/lib/module/adapter/db.js +96 -0
  65. package/lib/module/adapter/db.js.map +1 -0
  66. package/lib/module/adapter/index.js +9 -0
  67. package/lib/module/adapter/index.js.map +1 -0
  68. package/lib/module/adapter/storage.js +45 -0
  69. package/lib/module/adapter/storage.js.map +1 -0
  70. package/lib/module/auth/functions.js +3 -0
  71. package/lib/module/auth/functions.js.map +1 -0
  72. package/lib/module/auth/index.js +2 -0
  73. package/lib/module/auth/index.js.map +1 -0
  74. package/lib/module/components/AuthenticatorVerify.js +82 -0
  75. package/lib/module/components/AuthenticatorVerify.js.map +1 -0
  76. package/lib/module/components/FirebaseSignIn.js +187 -0
  77. package/lib/module/components/FirebaseSignIn.js.map +1 -0
  78. package/lib/module/components/PhoneVerify.js +116 -0
  79. package/lib/module/components/PhoneVerify.js.map +1 -0
  80. package/lib/module/components/TwoFactorAuthModal.js +110 -0
  81. package/lib/module/components/TwoFactorAuthModal.js.map +1 -0
  82. package/lib/module/components/index.js +3 -0
  83. package/lib/module/components/index.js.map +1 -0
  84. package/lib/module/contexts/FirebaseContext.js +39 -0
  85. package/lib/module/contexts/FirebaseContext.js.map +1 -0
  86. package/lib/module/contexts/index.js +2 -0
  87. package/lib/module/contexts/index.js.map +1 -0
  88. package/lib/module/db/index.js +2 -0
  89. package/lib/module/db/index.js.map +1 -0
  90. package/lib/module/db/utils.js +111 -0
  91. package/lib/module/db/utils.js.map +1 -0
  92. package/lib/module/hooks/backend.js +5 -0
  93. package/lib/module/hooks/backend.js.map +1 -0
  94. package/lib/module/hooks/index.js +2 -0
  95. package/lib/module/hooks/index.js.map +1 -0
  96. package/lib/module/hooks/useAuthenticatorVerify.js +45 -0
  97. package/lib/module/hooks/useAuthenticatorVerify.js.map +1 -0
  98. package/lib/module/hooks/usePhoneVerify.js +76 -0
  99. package/lib/module/hooks/usePhoneVerify.js.map +1 -0
  100. package/lib/module/icons/Google.js +22 -0
  101. package/lib/module/icons/Google.js.map +1 -0
  102. package/lib/module/index.js +10 -0
  103. package/lib/module/index.js.map +1 -0
  104. package/lib/module/storage/index.js +2 -0
  105. package/lib/module/storage/index.js.map +1 -0
  106. package/lib/module/storage/utils.js +30 -0
  107. package/lib/module/storage/utils.js.map +1 -0
  108. package/lib/module/types/adapter.js +2 -0
  109. package/lib/module/types/adapter.js.map +1 -0
  110. package/lib/module/types/auth.js +2 -0
  111. package/lib/module/types/auth.js.map +1 -0
  112. package/lib/module/types/db.js +2 -0
  113. package/lib/module/types/db.js.map +1 -0
  114. package/lib/module/types/functions.js +2 -0
  115. package/lib/module/types/functions.js.map +1 -0
  116. package/lib/module/types/index.js +2 -0
  117. package/lib/module/types/index.js.map +1 -0
  118. package/lib/module/types/storage.js +2 -0
  119. package/lib/module/types/storage.js.map +1 -0
  120. package/lib/module/utils/validation.js +62 -0
  121. package/lib/module/utils/validation.js.map +1 -0
  122. package/lib/typescript/adapter/auth.d.ts +7 -0
  123. package/lib/typescript/adapter/auth.d.ts.map +1 -0
  124. package/lib/typescript/adapter/db.d.ts +5 -0
  125. package/lib/typescript/adapter/db.d.ts.map +1 -0
  126. package/lib/typescript/adapter/index.d.ts +9 -0
  127. package/lib/typescript/adapter/index.d.ts.map +1 -0
  128. package/lib/typescript/adapter/storage.d.ts +4 -0
  129. package/lib/typescript/adapter/storage.d.ts.map +1 -0
  130. package/lib/typescript/auth/functions.d.ts +4 -0
  131. package/lib/typescript/auth/functions.d.ts.map +1 -0
  132. package/lib/typescript/auth/index.d.ts +2 -0
  133. package/lib/typescript/auth/index.d.ts.map +1 -0
  134. package/lib/typescript/components/AuthenticatorVerify.d.ts +3 -0
  135. package/lib/typescript/components/AuthenticatorVerify.d.ts.map +1 -0
  136. package/lib/typescript/components/FirebaseSignIn.d.ts +6 -0
  137. package/lib/typescript/components/FirebaseSignIn.d.ts.map +1 -0
  138. package/lib/typescript/components/PhoneVerify.d.ts +6 -0
  139. package/lib/typescript/components/PhoneVerify.d.ts.map +1 -0
  140. package/lib/typescript/components/TwoFactorAuthModal.d.ts +3 -0
  141. package/lib/typescript/components/TwoFactorAuthModal.d.ts.map +1 -0
  142. package/lib/typescript/components/index.d.ts +3 -0
  143. package/lib/typescript/components/index.d.ts.map +1 -0
  144. package/lib/typescript/contexts/FirebaseContext.d.ts +9 -0
  145. package/lib/typescript/contexts/FirebaseContext.d.ts.map +1 -0
  146. package/lib/typescript/contexts/index.d.ts +2 -0
  147. package/lib/typescript/contexts/index.d.ts.map +1 -0
  148. package/lib/typescript/db/index.d.ts +2 -0
  149. package/lib/typescript/db/index.d.ts.map +1 -0
  150. package/lib/typescript/db/utils.d.ts +6 -0
  151. package/lib/typescript/db/utils.d.ts.map +1 -0
  152. package/lib/typescript/hooks/backend.d.ts +2 -0
  153. package/lib/typescript/hooks/backend.d.ts.map +1 -0
  154. package/lib/typescript/hooks/index.d.ts +2 -0
  155. package/lib/typescript/hooks/index.d.ts.map +1 -0
  156. package/lib/typescript/hooks/useAuthenticatorVerify.d.ts +8 -0
  157. package/lib/typescript/hooks/useAuthenticatorVerify.d.ts.map +1 -0
  158. package/lib/typescript/hooks/usePhoneVerify.d.ts +9 -0
  159. package/lib/typescript/hooks/usePhoneVerify.d.ts.map +1 -0
  160. package/lib/typescript/icons/Google.d.ts +5 -0
  161. package/lib/typescript/icons/Google.d.ts.map +1 -0
  162. package/lib/typescript/index.d.ts +10 -0
  163. package/lib/typescript/index.d.ts.map +1 -0
  164. package/lib/typescript/storage/index.d.ts +2 -0
  165. package/lib/typescript/storage/index.d.ts.map +1 -0
  166. package/lib/typescript/storage/utils.d.ts +4 -0
  167. package/lib/typescript/storage/utils.d.ts.map +1 -0
  168. package/lib/typescript/types/adapter.d.ts +6 -0
  169. package/lib/typescript/types/adapter.d.ts.map +1 -0
  170. package/lib/typescript/types/auth.d.ts +12 -0
  171. package/lib/typescript/types/auth.d.ts.map +1 -0
  172. package/lib/typescript/types/db.d.ts +8 -0
  173. package/lib/typescript/types/db.d.ts.map +1 -0
  174. package/lib/typescript/types/functions.d.ts +3 -0
  175. package/lib/typescript/types/functions.d.ts.map +1 -0
  176. package/lib/typescript/types/index.d.ts +24 -0
  177. package/lib/typescript/types/index.d.ts.map +1 -0
  178. package/lib/typescript/types/storage.d.ts +3 -0
  179. package/lib/typescript/types/storage.d.ts.map +1 -0
  180. package/lib/typescript/utils/validation.d.ts +21 -0
  181. package/lib/typescript/utils/validation.d.ts.map +1 -0
  182. package/package.json +29 -12
  183. package/src/adapter/auth.ts +474 -0
  184. package/src/adapter/db.ts +146 -0
  185. package/src/adapter/index.ts +30 -0
  186. package/src/adapter/storage.ts +58 -0
  187. package/src/auth/functions.ts +7 -0
  188. package/src/auth/index.ts +1 -0
  189. package/src/components/AuthenticatorVerify.tsx +75 -0
  190. package/src/components/FirebaseSignIn.tsx +187 -0
  191. package/src/components/PhoneVerify.tsx +102 -0
  192. package/src/components/TwoFactorAuthModal.tsx +133 -0
  193. package/src/components/index.ts +2 -0
  194. package/src/contexts/FirebaseContext.tsx +54 -0
  195. package/src/contexts/index.ts +1 -0
  196. package/src/db/index.ts +1 -0
  197. package/src/db/utils.ts +142 -0
  198. package/src/hooks/backend.ts +4 -0
  199. package/src/hooks/index.ts +1 -0
  200. package/src/hooks/useAuthenticatorVerify.ts +45 -0
  201. package/src/hooks/usePhoneVerify.ts +76 -0
  202. package/src/icons/Google.tsx +24 -0
  203. package/src/index.ts +9 -0
  204. package/src/storage/index.ts +1 -0
  205. package/src/storage/utils.ts +29 -0
  206. package/src/types/adapter.ts +13 -0
  207. package/src/types/auth.ts +13 -0
  208. package/src/types/db.ts +10 -0
  209. package/src/types/functions.ts +3 -0
  210. package/src/types/index.ts +26 -0
  211. package/src/types/storage.ts +3 -0
  212. package/src/utils/validation.ts +85 -0
@@ -0,0 +1,474 @@
1
+ import {
2
+ AuthAdapter,
3
+ BaseAuthProvider,
4
+ BaseUserData,
5
+ BaseUserRole,
6
+ EnrollmentFactor,
7
+ EnrollmentFactorsResult,
8
+ LoginResult,
9
+ MultiFactorVerification,
10
+ WithMultiFactorVerified,
11
+ } from '@chem-po/core'
12
+ import {
13
+ FirebaseAuthTypes,
14
+ getIdTokenResult,
15
+ getMultiFactorResolver,
16
+ GoogleAuthProvider,
17
+ onAuthStateChanged,
18
+ PhoneAuthProvider,
19
+ PhoneMultiFactorGenerator,
20
+ sendPasswordResetEmail,
21
+ signInWithCredential,
22
+ signInWithCustomToken,
23
+ signInWithEmailAndPassword,
24
+ signOut,
25
+ } from '@react-native-firebase/auth'
26
+
27
+ import { GoogleSignin } from '@react-native-google-signin/google-signin'
28
+ import { Auth, User } from '../types/auth'
29
+
30
+ const isDebug = (
31
+ typeof process !== 'undefined' &&
32
+ process.env?.EXPO_PUBLIC_DEBUG === 'true'
33
+ ) || false
34
+
35
+ const debugLog = (message: string, data?: any) => {
36
+ if (isDebug) {
37
+ console.log(`[FirebaseAuth Debug] ${message}`, data || '')
38
+ }
39
+ }
40
+
41
+ const providerInitialized: Record<string, boolean> = {}
42
+
43
+ const getUserWithRole = async <UserData extends BaseUserData>(
44
+ user: FirebaseAuthTypes.User
45
+ ): Promise<WithMultiFactorVerified<UserData>> => {
46
+ debugLog('Getting user with role', { uid: user.uid, email: user.email })
47
+
48
+ try {
49
+ const { claims } = await getIdTokenResult(user)
50
+ debugLog('Retrieved ID token claims', { role: claims.role, customClaims: Object.keys(claims) })
51
+
52
+ const userWithRole = {
53
+ ...(user as any),
54
+ role: claims.role ?? ('user' as BaseUserRole),
55
+ multiFactorVerified: !!user.multiFactor?.enrolledFactors.length
56
+ } as WithMultiFactorVerified<UserData>
57
+
58
+ debugLog('User with role created', {
59
+ uid: userWithRole.uid,
60
+ role: userWithRole.role,
61
+ multiFactorVerified: userWithRole.multiFactorVerified,
62
+ enrolledFactorsCount: user.multiFactor?.enrolledFactors.length || 0
63
+ })
64
+
65
+ return userWithRole
66
+ } catch (error) {
67
+ debugLog('Error getting user with role', error)
68
+ throw error
69
+ }
70
+ }
71
+
72
+ const toEnrollmentFactor = (hint: FirebaseAuthTypes.MultiFactorInfo): EnrollmentFactor => {
73
+ if (hint.factorId === PhoneMultiFactorGenerator.FACTOR_ID) {
74
+ return {
75
+ type: 'phone',
76
+ phoneNumber: (hint as FirebaseAuthTypes.PhoneMultiFactorInfo).phoneNumber,
77
+ enrollmentTime: hint.enrollmentTime,
78
+ displayName: (hint as FirebaseAuthTypes.PhoneMultiFactorInfo).displayName,
79
+ uid: hint.uid,
80
+ }
81
+ }
82
+ if (hint.factorId === 'totp') {
83
+ return {
84
+ type: 'totp',
85
+ enrollmentTime: hint.enrollmentTime,
86
+ displayName: hint.displayName,
87
+ uid: hint.uid,
88
+ }
89
+ }
90
+ throw new Error('Unsupported factor type: ' + hint.factorId)
91
+ }
92
+
93
+ const toFirebaseFactor = (factor: EnrollmentFactor): FirebaseAuthTypes.MultiFactorInfo => {
94
+ if (factor.type === 'phone') {
95
+ return {
96
+ factorId: PhoneMultiFactorGenerator.FACTOR_ID,
97
+ phoneNumber: factor.phoneNumber,
98
+ enrollmentTime: factor.enrollmentTime,
99
+ uid: factor.uid,
100
+ displayName: factor.displayName,
101
+ }
102
+ }
103
+ if (factor.type === 'totp') {
104
+ return {
105
+ factorId: 'totp',
106
+ enrollmentTime: factor.enrollmentTime,
107
+ uid: factor.uid,
108
+ displayName: factor.displayName,
109
+ }
110
+ }
111
+ throw new Error(
112
+ `Unsupported factor type: ${(factor as FirebaseAuthTypes.MultiFactorInfo).factorId ?? 'Missing factor type'}`
113
+ )
114
+ }
115
+
116
+ const sendMultiFactorCode = async (
117
+ auth: Auth,
118
+ factor: EnrollmentFactor,
119
+ resolver: FirebaseAuthTypes.MultiFactorResolver
120
+ ): Promise<MultiFactorVerification> => {
121
+ debugLog('Sending multi-factor code', { factorType: factor.type, factorUid: factor.uid })
122
+
123
+ const sessionId = resolver.session
124
+ // const verificationId = await new PhoneAuthProvider(auth).verifyPhoneNumber(phoneSignInFactor, sessionId)
125
+ if (factor.type === 'phone') {
126
+ try {
127
+ const verificationId = await auth.verifyPhoneNumberWithMultiFactorInfo(
128
+ toFirebaseFactor(factor),
129
+ sessionId
130
+ )
131
+ debugLog('Multi-factor code sent successfully', { verificationId: verificationId.substring(0, 10) + '...' })
132
+ return {
133
+ verificationId,
134
+ factor,
135
+ resolver,
136
+ }
137
+ } catch (error) {
138
+ debugLog('Error sending multi-factor code', error)
139
+ throw error
140
+ }
141
+ }
142
+ throw new Error(`Unsupported factor type: ${factor.type ?? 'Missing factor type'}`)
143
+ }
144
+
145
+ const getEnrolledFactors = async (auth: Auth, error: any): Promise<EnrollmentFactorsResult> => {
146
+ debugLog('Getting enrolled factors from error', { errorCode: error?.code })
147
+
148
+ try {
149
+ const resolver = getMultiFactorResolver(auth, error)
150
+ debugLog('Multi-factor resolver created', { hintsCount: resolver.hints.length })
151
+
152
+ if (resolver.hints.length === 0) {
153
+ debugLog('No enrolled factors found')
154
+ throw new Error('No multi-factor verification methods found, please enroll one on the website')
155
+ }
156
+
157
+ const enrollmentFactors = resolver.hints.map(toEnrollmentFactor)
158
+ debugLog('Enrollment factors mapped', { factors: enrollmentFactors.map(f => ({ type: f.type, uid: f.uid })) })
159
+
160
+ return {
161
+ enrollmentFactors,
162
+ multiFactorResolver: resolver,
163
+ }
164
+ } catch (error) {
165
+ debugLog('Error getting enrolled factors', error)
166
+ throw error
167
+ }
168
+ }
169
+
170
+ const verifyMultiFactor = async (
171
+ verification: MultiFactorVerification,
172
+ code: string
173
+ ): Promise<LoginResult<User>> => {
174
+ debugLog('Verifying multi-factor code', {
175
+ factorType: verification.factor.type,
176
+ codeLength: code.length
177
+ })
178
+
179
+ try {
180
+ const credential = await PhoneAuthProvider.credential(verification.verificationId, code)
181
+ debugLog('Phone credential created')
182
+
183
+ const assertion = PhoneMultiFactorGenerator.assertion(credential)
184
+ debugLog('Multi-factor assertion created')
185
+
186
+ const resolver = (verification as any).resolver
187
+ if (!resolver) {
188
+ debugLog('Multi-factor resolver not found')
189
+ throw new Error('Internal error signing in with two factor: resolver not found')
190
+ }
191
+
192
+ const userCredential = await resolver.resolveSignIn(assertion)
193
+ debugLog('Multi-factor sign-in resolved', { uid: userCredential.user?.uid })
194
+
195
+ const user = userCredential.user
196
+ if (!user) {
197
+ debugLog('No user found after multi-factor resolution')
198
+ throw new Error('No user found')
199
+ }
200
+
201
+ const userWithRole = await getUserWithRole(user)
202
+ debugLog('Multi-factor verification completed successfully')
203
+
204
+ return { user: userWithRole }
205
+ } catch (error) {
206
+ debugLog('Error verifying multi-factor', error)
207
+ throw error
208
+ }
209
+ }
210
+
211
+ const handleSignInError = async (error: any): Promise<LoginResult<User>> => {
212
+ debugLog('Handling sign-in error', { code: error?.code, message: error?.message })
213
+
214
+ if (error.code === 'auth/multi-factor-auth-required') {
215
+ debugLog('Multi-factor authentication required')
216
+ return { requestArgs: error }
217
+ }
218
+
219
+ debugLog('Re-throwing sign-in error')
220
+ throw error
221
+ }
222
+
223
+ const initializeProvider = async (provider: BaseAuthProvider) => {
224
+ debugLog('Initializing provider', { name: provider.name, alreadyInitialized: providerInitialized[provider.name] })
225
+
226
+ if (providerInitialized[provider.name]) {
227
+ debugLog('Provider already initialized, skipping')
228
+ return
229
+ }
230
+
231
+ switch (provider.name) {
232
+ case 'google':
233
+ debugLog('Initializing Google provider', { webClientId: !!(provider as GoogleAuthProvider).webClientId })
234
+
235
+ if (!(provider as GoogleAuthProvider).webClientId) {
236
+ debugLog('Google web client ID missing')
237
+ throw new Error(
238
+ 'Google web client ID is required when using Google Auth. ' +
239
+ 'Get your webClientId from Firebase Console > Authentication > Sign-in method > Google > Web SDK configuration. ' +
240
+ 'Then provide it in your GoogleAuthProvider: { name: "google", webClientId: "YOUR_CLIENT_ID" }'
241
+ )
242
+ }
243
+
244
+ try {
245
+ await GoogleSignin.hasPlayServices({ showPlayServicesUpdateDialog: true })
246
+ debugLog('Google Play Services available')
247
+
248
+ GoogleSignin.configure({
249
+ webClientId: (provider as any).webClientId,
250
+ })
251
+ debugLog('Google Sign-In configured successfully')
252
+ } catch (error) {
253
+ debugLog('Error initializing Google provider', error)
254
+ if (error instanceof Error && error.message.includes('UNAVAILABLE')) {
255
+ throw new Error(
256
+ 'Google Play Services is not available or needs to be updated. ' +
257
+ 'Please ensure Google Play Services is installed and up to date on this device.'
258
+ )
259
+ }
260
+ throw error
261
+ }
262
+ break
263
+ case 'email':
264
+ debugLog('Email provider initialization (no special setup required)')
265
+ // Native SDK doesn't need special initialization
266
+ break
267
+ default:
268
+ debugLog('Unsupported provider', { name: provider.name })
269
+ throw new Error(`Unsupported provider: ${provider.name}`)
270
+ }
271
+
272
+ providerInitialized[provider.name] = true
273
+ debugLog('Provider initialization completed', { name: provider.name })
274
+ }
275
+
276
+ const handleInitialLogin = async (
277
+ userCredential: FirebaseAuthTypes.UserCredential,
278
+ twoFactorRequired: boolean
279
+ ): Promise<LoginResult<User>> => {
280
+ debugLog('Handling initial login', {
281
+ uid: userCredential.user?.uid,
282
+ twoFactorRequired,
283
+ enrolledFactorsCount: userCredential.user?.multiFactor?.enrolledFactors.length || 0
284
+ })
285
+
286
+ const user = userCredential.user
287
+ if (!user) {
288
+ debugLog('No user found in credential')
289
+ throw new Error('No user found')
290
+ }
291
+
292
+ if (twoFactorRequired) {
293
+ debugLog('Two-factor authentication is required but user has no enrolled factors')
294
+ throw new Error(
295
+ 'This app requires two factor authentication, please enroll a factor on the website and try again'
296
+ )
297
+ }
298
+
299
+ const userWithRole = await getUserWithRole(user)
300
+ debugLog('Initial login completed successfully')
301
+
302
+ return { user: userWithRole }
303
+ }
304
+
305
+ const loginWithGoogle = async (
306
+ auth: Auth,
307
+ twoFactorRequired: boolean
308
+ ): Promise<LoginResult<User>> => {
309
+ debugLog('Starting Google sign-in', { twoFactorRequired })
310
+
311
+ try {
312
+ debugLog('Calling GoogleSignin.signIn()')
313
+ const signInResult = await GoogleSignin.signIn()
314
+ debugLog('Google sign-in result received', { hasIdToken: !!signInResult.data?.idToken })
315
+
316
+ const idToken = signInResult.data?.idToken
317
+ if (!idToken) {
318
+ debugLog('No ID token found in Google sign-in result')
319
+ throw new Error('No ID token found')
320
+ }
321
+
322
+ debugLog('Creating Google credential')
323
+ const googleCredential = GoogleAuthProvider.credential(idToken)
324
+
325
+ debugLog('Signing in with Google credential')
326
+ const userCredential = await signInWithCredential(auth, googleCredential)
327
+
328
+ return await handleInitialLogin(userCredential, twoFactorRequired)
329
+ } catch (error) {
330
+ debugLog('Error in Google sign-in', error)
331
+ return await handleSignInError(error)
332
+ }
333
+ }
334
+
335
+ const getLoginWithPassword =
336
+ (auth: Auth, twoFactorRequired: boolean) =>
337
+ async (
338
+ provider: BaseAuthProvider,
339
+ { email, password }: { email: string; password: string }
340
+ ): Promise<LoginResult<User>> => {
341
+ debugLog('Starting email/password sign-in', {
342
+ provider: provider.name,
343
+ email,
344
+ twoFactorRequired
345
+ })
346
+
347
+ await initializeProvider(provider)
348
+ try {
349
+ debugLog('Calling signInWithEmailAndPassword')
350
+ const userCredential = await signInWithEmailAndPassword(auth, email, password)
351
+ debugLog('Email/password sign-in successful', { uid: userCredential.user?.uid })
352
+
353
+ return await handleInitialLogin(userCredential, twoFactorRequired)
354
+ } catch (error) {
355
+ debugLog('Error in email/password sign-in', error)
356
+ return await handleSignInError(error)
357
+ }
358
+ }
359
+
360
+ const getLoginWithPopup =
361
+ (auth: Auth, twoFactorRequired: boolean) =>
362
+ async (provider: BaseAuthProvider): Promise<LoginResult<User>> => {
363
+ debugLog('Starting popup sign-in', { provider: provider.name, twoFactorRequired })
364
+
365
+ await initializeProvider(provider)
366
+ switch (provider.name) {
367
+ case 'google':
368
+ return loginWithGoogle(auth, twoFactorRequired)
369
+ default:
370
+ debugLog('Unsupported popup provider', { name: provider.name })
371
+ throw new Error(`Unsupported provider: ${provider.name}`)
372
+ }
373
+ }
374
+
375
+ const getLoginWithToken =
376
+ (auth: Auth, twoFactorRequired: boolean) =>
377
+ async (provider: BaseAuthProvider, token: string): Promise<LoginResult<User>> => {
378
+ debugLog('Starting token sign-in', {
379
+ provider: provider.name,
380
+ tokenLength: token.length,
381
+ twoFactorRequired
382
+ })
383
+
384
+ await initializeProvider(provider)
385
+ try {
386
+ debugLog('Calling signInWithCustomToken')
387
+ const userCredential = await signInWithCustomToken(auth, token)
388
+ debugLog('Token sign-in successful', { uid: userCredential.user?.uid })
389
+
390
+ return await handleInitialLogin(userCredential, twoFactorRequired)
391
+ } catch (error) {
392
+ debugLog('Error in token sign-in', error)
393
+ return await handleSignInError(error)
394
+ }
395
+ }
396
+
397
+ const resetPassword = async (
398
+ auth: Auth,
399
+ provider: BaseAuthProvider,
400
+ usernameOrEmail: string
401
+ ): Promise<void> => {
402
+ debugLog('Starting password reset', { provider: provider.name, email: usernameOrEmail })
403
+
404
+ await initializeProvider(provider)
405
+ try {
406
+ await sendPasswordResetEmail(auth, usernameOrEmail)
407
+ debugLog('Password reset email sent successfully')
408
+ } catch (error) {
409
+ debugLog('Error sending password reset email', error)
410
+ throw error
411
+ }
412
+ }
413
+
414
+ export const getFirebaseAuthAdapter = (
415
+ auth: Auth,
416
+ twoFactorRequired: boolean
417
+ ): AuthAdapter<BaseAuthProvider, User, { email: string; password: string }> => {
418
+ debugLog('Creating Firebase Auth Adapter', { twoFactorRequired })
419
+
420
+ return {
421
+ twoFactorRequired,
422
+ getCurrentUser: async () => {
423
+ debugLog('Getting current user')
424
+ const user = auth.currentUser
425
+ if (!user) {
426
+ debugLog('No current user found')
427
+ return null
428
+ }
429
+ debugLog('Current user found', { uid: user.uid, email: user.email })
430
+ return getUserWithRole(user)
431
+ },
432
+ loginWithPassword: getLoginWithPassword(auth, twoFactorRequired),
433
+ loginWithPopup: getLoginWithPopup(auth, twoFactorRequired),
434
+ loginWithToken: getLoginWithToken(auth, twoFactorRequired),
435
+ resetPassword: (...args) => {
436
+ return resetPassword(auth, ...args)
437
+ },
438
+ verifyMultiFactor,
439
+ sendMultiFactorCode: (...args) => {
440
+ return sendMultiFactorCode(auth, ...args)
441
+ },
442
+ getEnrolledFactors: (...args) => {
443
+ return getEnrolledFactors(auth, ...args)
444
+ },
445
+ logout: async () => {
446
+ debugLog('Starting logout')
447
+ try {
448
+ await signOut(auth)
449
+ debugLog('Logout completed successfully')
450
+ } catch (error) {
451
+ debugLog('Error during logout', error)
452
+ throw error
453
+ }
454
+ },
455
+ subscribeToUser: (callback) => {
456
+ debugLog('Setting up user subscription')
457
+ return onAuthStateChanged(auth, (user) => {
458
+ if (!user) {
459
+ debugLog('User state changed: signed out')
460
+ callback(null)
461
+ return
462
+ }
463
+ debugLog('User state changed: user signed in', { uid: user.uid, email: user.email })
464
+ getUserWithRole(user).then((result) => {
465
+ debugLog('User subscription callback completed', { uid: result.uid, role: result.role })
466
+ callback(result)
467
+ }).catch((error) => {
468
+ debugLog('Error in user subscription callback', error)
469
+ callback(null)
470
+ })
471
+ })
472
+ },
473
+ }
474
+ }
@@ -0,0 +1,146 @@
1
+ import {
2
+ addMetadata,
3
+ AnyObject,
4
+ BaseError,
5
+ DatabaseAdapter,
6
+ DBItem,
7
+ OnError,
8
+ OnItemData,
9
+ Unsubscribe,
10
+ WithMetadata,
11
+ } from '@chem-po/core'
12
+ import { useAuth } from '@chem-po/react'
13
+ import {
14
+ deleteDoc,
15
+ doc,
16
+ FirebaseFirestoreTypes,
17
+ getDoc,
18
+ getDocs,
19
+ onSnapshot,
20
+ setDoc,
21
+ } from '@react-native-firebase/firestore'
22
+ import { httpsCallable } from '@react-native-firebase/functions'
23
+ import { toCursorQuery, toFirestoreQuery } from '../db/utils'
24
+ import { Firestore, FirestoreBaseQuery, FirestoreCursor, FirestoreOnItemsData } from '../types/db'
25
+ import { Functions } from '../types/functions'
26
+ // import { BaseQuery, FirestoreCursor } from '../types/db'
27
+
28
+ const handleFirestoreError = (error: any): BaseError => ({
29
+ code: error?.code ?? 'unknown',
30
+ message: error?.name ?? 'Unknown Error',
31
+ description: error?.message ?? 'An unknown error occurred',
32
+ })
33
+
34
+ const toDBItem = <T extends AnyObject>(
35
+ doc: FirebaseFirestoreTypes.DocumentSnapshot<T>,
36
+ ): DBItem<T> => ({
37
+ ...(doc.data() as WithMetadata<T>),
38
+ id: doc.id,
39
+ })
40
+
41
+ const getSubscribeToQuery =
42
+ (db: Firestore) =>
43
+ <T extends AnyObject>(
44
+ baseQuery: FirestoreBaseQuery<T>,
45
+ onData: FirestoreOnItemsData<T>,
46
+ onError: OnError,
47
+ ): Unsubscribe => {
48
+ const q = toFirestoreQuery<T>(db, baseQuery)
49
+ return onSnapshot(
50
+ q,
51
+ snapshot => {
52
+ const items = snapshot.docs.map(doc => toDBItem(doc))
53
+ onData(items, snapshot.docs[snapshot.docs.length - 1] as FirestoreCursor)
54
+ },
55
+ error => onError(handleFirestoreError(error)),
56
+ )
57
+ }
58
+
59
+ const getFetchNextCursor =
60
+ (db: Firestore) =>
61
+ async <T extends AnyObject>(
62
+ baseQuery: FirestoreBaseQuery<T>,
63
+ ): Promise<FirestoreCursor | null> => {
64
+ const { cursor } = baseQuery
65
+ const cursorDoc = cursor ? await getDoc(doc(db, baseQuery.collection, cursor.id)) : null
66
+ const q = toCursorQuery<T>(db, baseQuery, cursorDoc as FirestoreCursor | null)
67
+ const snapshot = await getDocs(q)
68
+ return (snapshot.docs[0] ?? null) as FirestoreCursor | null
69
+ }
70
+
71
+ const getFetchCount =
72
+ (functions: Functions) =>
73
+ async <T extends AnyObject>(baseQuery: FirestoreBaseQuery<T>): Promise<number> => {
74
+ const fetchCountFunc = httpsCallable(functions, 'getQueryCount')
75
+ const { data } = await fetchCountFunc(baseQuery)
76
+ return (data as { count: number }).count
77
+ }
78
+
79
+ const getFetchItem =
80
+ (db: Firestore) =>
81
+ async <T extends AnyObject>(path: string): Promise<DBItem<T> | null> => {
82
+ const docRef = doc(db, path)
83
+ const docSnap = await getDoc(docRef)
84
+ const data = docSnap.data()
85
+ return data ? ({ ...data, id: docSnap.id } as DBItem<T>) : null
86
+ }
87
+
88
+ const getFetchItems =
89
+ (db: Firestore) =>
90
+ async <T extends AnyObject>(baseQuery: FirestoreBaseQuery<T>): Promise<DBItem<T>[]> => {
91
+ const q = toFirestoreQuery<T>(db, baseQuery)
92
+ const snapshot = await getDocs(q)
93
+ return snapshot.docs.map(doc => ({ ...doc.data(), id: doc.id }) as DBItem<T>)
94
+ }
95
+
96
+ const getCreateItem =
97
+ (db: Firestore) =>
98
+ async (path: string, item: AnyObject): Promise<string> => {
99
+ const docRef = doc(db, path)
100
+ const data = addMetadata(useAuth.getState().user, item, true)
101
+ await setDoc(docRef, data, { merge: true })
102
+ return docRef.id
103
+ }
104
+
105
+ const getDeleteItem =
106
+ (db: Firestore) =>
107
+ async (path: string): Promise<void> => {
108
+ const docRef = doc(db, path)
109
+ await deleteDoc(docRef)
110
+ }
111
+
112
+ const getUpdateItem =
113
+ (db: Firestore) =>
114
+ async <T extends AnyObject>(path: string, item: Partial<T>): Promise<void> => {
115
+ const docRef = doc(db, path)
116
+ await setDoc(docRef, item, { merge: true })
117
+ }
118
+
119
+ const getSubscribeToItem =
120
+ (db: Firestore) =>
121
+ <T extends AnyObject>(path: string, onData: OnItemData<T>, onError: OnError): Unsubscribe => {
122
+ const docRef = doc(db, path)
123
+ return onSnapshot(
124
+ docRef,
125
+ snapshot => {
126
+ const data = snapshot.data()
127
+ onData(data ? ({ ...data, id: snapshot.id } as DBItem<T>) : null)
128
+ },
129
+ error => onError(handleFirestoreError(error)),
130
+ )
131
+ }
132
+
133
+ export const getFirebaseDatabaseAdapter = (
134
+ db: Firestore,
135
+ functions: Functions,
136
+ ): DatabaseAdapter<FirestoreCursor> => ({
137
+ fetchNextCursor: getFetchNextCursor(db),
138
+ fetchCount: getFetchCount(functions),
139
+ fetchItem: getFetchItem(db),
140
+ fetchItems: getFetchItems(db),
141
+ createItem: getCreateItem(db),
142
+ deleteItem: getDeleteItem(db),
143
+ updateItem: getUpdateItem(db),
144
+ subscribeToItem: getSubscribeToItem(db),
145
+ subscribeToQuery: getSubscribeToQuery(db),
146
+ })
@@ -0,0 +1,30 @@
1
+ import { BackendAdapter, BaseAuthProvider } from '@chem-po/core'
2
+ import { FirebaseStorageTypes } from '@react-native-firebase/storage'
3
+ import { Auth, EmailPasswordLogin, User } from '../types/auth'
4
+ import { Firestore, FirestoreCursor } from '../types/db'
5
+ import { Functions } from '../types/functions'
6
+ import { Storage } from '../types/storage'
7
+ import { getFirebaseAuthAdapter } from './auth'
8
+ import { getFirebaseDatabaseAdapter } from './db'
9
+ import { getFirebaseStorageAdapter } from './storage'
10
+
11
+ export type FirebaseAdapter<AuthProvider extends BaseAuthProvider> = BackendAdapter<
12
+ AuthProvider,
13
+ User,
14
+ EmailPasswordLogin,
15
+ FirestoreCursor,
16
+ Blob,
17
+ FirebaseStorageTypes.FullMetadata
18
+ >
19
+
20
+ export const getFirebaseAdapter = <AuthProvider extends BaseAuthProvider>(
21
+ auth: Auth,
22
+ db: Firestore,
23
+ storage: Storage,
24
+ functions: Functions,
25
+ twoFactorRequired: boolean,
26
+ ): FirebaseAdapter<AuthProvider> => ({
27
+ auth: getFirebaseAuthAdapter(auth, twoFactorRequired),
28
+ db: getFirebaseDatabaseAdapter(db, functions),
29
+ storage: getFirebaseStorageAdapter(storage),
30
+ })
@@ -0,0 +1,58 @@
1
+ import { StorageAdapter, UploadedFileValue } from '@chem-po/core'
2
+ import {
3
+ deleteObject,
4
+ getDownloadURL,
5
+ getMetadata,
6
+ ref,
7
+ uploadBytesResumable,
8
+ } from '@react-native-firebase/storage'
9
+ import { Storage } from '../types/storage'
10
+
11
+ export const getFirebaseStorageAdapter = (storage: Storage): StorageAdapter<Blob, any> => ({
12
+ delete: async path => {
13
+ const storageRef = ref(storage, path)
14
+ await deleteObject(storageRef)
15
+ },
16
+ fetchObject: async path => {
17
+ const storageRef = ref(storage, path)
18
+ const url = await getDownloadURL(storageRef)
19
+ const response = await fetch(url)
20
+ return await response.blob()
21
+ },
22
+ fetchMetadata: async path => {
23
+ const storageRef = ref(storage, path)
24
+ return await getMetadata(storageRef)
25
+ },
26
+ getObjectUrl: async path => {
27
+ const storageRef = ref(storage, path)
28
+ return await getDownloadURL(storageRef)
29
+ },
30
+ upload: async (path, data, onUploadProgress) => {
31
+ const storageRef = ref(storage, path)
32
+ // Convert dataUrl to Blob
33
+ const blob = await fetch(data.dataUrl).then(res => res.blob())
34
+ const uploadTask = uploadBytesResumable(storageRef, blob)
35
+ return new Promise<UploadedFileValue>((resolve, reject) => {
36
+ uploadTask.on(
37
+ 'state_changed',
38
+ snapshot => {
39
+ const progress = (snapshot.bytesTransferred / snapshot.totalBytes) * 100
40
+ onUploadProgress({
41
+ loaded: snapshot.bytesTransferred,
42
+ total: snapshot.totalBytes,
43
+ percent: progress,
44
+ })
45
+ },
46
+ error => reject(error),
47
+ () => {
48
+ const fileValue: UploadedFileValue = {
49
+ filename: data.filename,
50
+ type: data.type,
51
+ storagePath: path,
52
+ }
53
+ resolve(fileValue)
54
+ },
55
+ )
56
+ })
57
+ },
58
+ })