@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.
- package/README.md +215 -0
- package/lib/commonjs/adapter/auth.js +431 -0
- package/lib/commonjs/adapter/auth.js.map +1 -0
- package/lib/commonjs/adapter/db.js +103 -0
- package/lib/commonjs/adapter/db.js.map +1 -0
- package/lib/commonjs/adapter/index.js +16 -0
- package/lib/commonjs/adapter/index.js.map +1 -0
- package/lib/commonjs/adapter/storage.js +52 -0
- package/lib/commonjs/adapter/storage.js.map +1 -0
- package/lib/commonjs/auth/functions.js +11 -0
- package/lib/commonjs/auth/functions.js.map +1 -0
- package/lib/commonjs/auth/index.js +17 -0
- package/lib/commonjs/auth/index.js.map +1 -0
- package/lib/commonjs/components/AuthenticatorVerify.js +90 -0
- package/lib/commonjs/components/AuthenticatorVerify.js.map +1 -0
- package/lib/commonjs/components/FirebaseSignIn.js +196 -0
- package/lib/commonjs/components/FirebaseSignIn.js.map +1 -0
- package/lib/commonjs/components/PhoneVerify.js +123 -0
- package/lib/commonjs/components/PhoneVerify.js.map +1 -0
- package/lib/commonjs/components/TwoFactorAuthModal.js +118 -0
- package/lib/commonjs/components/TwoFactorAuthModal.js.map +1 -0
- package/lib/commonjs/components/index.js +28 -0
- package/lib/commonjs/components/index.js.map +1 -0
- package/lib/commonjs/contexts/FirebaseContext.js +48 -0
- package/lib/commonjs/contexts/FirebaseContext.js.map +1 -0
- package/lib/commonjs/contexts/index.js +17 -0
- package/lib/commonjs/contexts/index.js.map +1 -0
- package/lib/commonjs/db/index.js +17 -0
- package/lib/commonjs/db/index.js.map +1 -0
- package/lib/commonjs/db/utils.js +120 -0
- package/lib/commonjs/db/utils.js.map +1 -0
- package/lib/commonjs/hooks/backend.js +12 -0
- package/lib/commonjs/hooks/backend.js.map +1 -0
- package/lib/commonjs/hooks/index.js +17 -0
- package/lib/commonjs/hooks/index.js.map +1 -0
- package/lib/commonjs/hooks/useAuthenticatorVerify.js +52 -0
- package/lib/commonjs/hooks/useAuthenticatorVerify.js.map +1 -0
- package/lib/commonjs/hooks/usePhoneVerify.js +83 -0
- package/lib/commonjs/hooks/usePhoneVerify.js.map +1 -0
- package/lib/commonjs/icons/Google.js +29 -0
- package/lib/commonjs/icons/Google.js.map +1 -0
- package/lib/commonjs/index.js +105 -0
- package/lib/commonjs/index.js.map +1 -0
- package/lib/commonjs/storage/index.js +17 -0
- package/lib/commonjs/storage/index.js.map +1 -0
- package/lib/commonjs/storage/utils.js +37 -0
- package/lib/commonjs/storage/utils.js.map +1 -0
- package/lib/commonjs/types/adapter.js +6 -0
- package/lib/commonjs/types/adapter.js.map +1 -0
- package/lib/commonjs/types/auth.js +6 -0
- package/lib/commonjs/types/auth.js.map +1 -0
- package/lib/commonjs/types/db.js +6 -0
- package/lib/commonjs/types/db.js.map +1 -0
- package/lib/commonjs/types/functions.js +6 -0
- package/lib/commonjs/types/functions.js.map +1 -0
- package/lib/commonjs/types/index.js +6 -0
- package/lib/commonjs/types/index.js.map +1 -0
- package/lib/commonjs/types/storage.js +6 -0
- package/lib/commonjs/types/storage.js.map +1 -0
- package/lib/commonjs/utils/validation.js +71 -0
- package/lib/commonjs/utils/validation.js.map +1 -0
- package/lib/module/adapter/auth.js +424 -0
- package/lib/module/adapter/auth.js.map +1 -0
- package/lib/module/adapter/db.js +96 -0
- package/lib/module/adapter/db.js.map +1 -0
- package/lib/module/adapter/index.js +9 -0
- package/lib/module/adapter/index.js.map +1 -0
- package/lib/module/adapter/storage.js +45 -0
- package/lib/module/adapter/storage.js.map +1 -0
- package/lib/module/auth/functions.js +3 -0
- package/lib/module/auth/functions.js.map +1 -0
- package/lib/module/auth/index.js +2 -0
- package/lib/module/auth/index.js.map +1 -0
- package/lib/module/components/AuthenticatorVerify.js +82 -0
- package/lib/module/components/AuthenticatorVerify.js.map +1 -0
- package/lib/module/components/FirebaseSignIn.js +187 -0
- package/lib/module/components/FirebaseSignIn.js.map +1 -0
- package/lib/module/components/PhoneVerify.js +116 -0
- package/lib/module/components/PhoneVerify.js.map +1 -0
- package/lib/module/components/TwoFactorAuthModal.js +110 -0
- package/lib/module/components/TwoFactorAuthModal.js.map +1 -0
- package/lib/module/components/index.js +3 -0
- package/lib/module/components/index.js.map +1 -0
- package/lib/module/contexts/FirebaseContext.js +39 -0
- package/lib/module/contexts/FirebaseContext.js.map +1 -0
- package/lib/module/contexts/index.js +2 -0
- package/lib/module/contexts/index.js.map +1 -0
- package/lib/module/db/index.js +2 -0
- package/lib/module/db/index.js.map +1 -0
- package/lib/module/db/utils.js +111 -0
- package/lib/module/db/utils.js.map +1 -0
- package/lib/module/hooks/backend.js +5 -0
- package/lib/module/hooks/backend.js.map +1 -0
- package/lib/module/hooks/index.js +2 -0
- package/lib/module/hooks/index.js.map +1 -0
- package/lib/module/hooks/useAuthenticatorVerify.js +45 -0
- package/lib/module/hooks/useAuthenticatorVerify.js.map +1 -0
- package/lib/module/hooks/usePhoneVerify.js +76 -0
- package/lib/module/hooks/usePhoneVerify.js.map +1 -0
- package/lib/module/icons/Google.js +22 -0
- package/lib/module/icons/Google.js.map +1 -0
- package/lib/module/index.js +10 -0
- package/lib/module/index.js.map +1 -0
- package/lib/module/storage/index.js +2 -0
- package/lib/module/storage/index.js.map +1 -0
- package/lib/module/storage/utils.js +30 -0
- package/lib/module/storage/utils.js.map +1 -0
- package/lib/module/types/adapter.js +2 -0
- package/lib/module/types/adapter.js.map +1 -0
- package/lib/module/types/auth.js +2 -0
- package/lib/module/types/auth.js.map +1 -0
- package/lib/module/types/db.js +2 -0
- package/lib/module/types/db.js.map +1 -0
- package/lib/module/types/functions.js +2 -0
- package/lib/module/types/functions.js.map +1 -0
- package/lib/module/types/index.js +2 -0
- package/lib/module/types/index.js.map +1 -0
- package/lib/module/types/storage.js +2 -0
- package/lib/module/types/storage.js.map +1 -0
- package/lib/module/utils/validation.js +62 -0
- package/lib/module/utils/validation.js.map +1 -0
- package/lib/typescript/adapter/auth.d.ts +7 -0
- package/lib/typescript/adapter/auth.d.ts.map +1 -0
- package/lib/typescript/adapter/db.d.ts +5 -0
- package/lib/typescript/adapter/db.d.ts.map +1 -0
- package/lib/typescript/adapter/index.d.ts +9 -0
- package/lib/typescript/adapter/index.d.ts.map +1 -0
- package/lib/typescript/adapter/storage.d.ts +4 -0
- package/lib/typescript/adapter/storage.d.ts.map +1 -0
- package/lib/typescript/auth/functions.d.ts +4 -0
- package/lib/typescript/auth/functions.d.ts.map +1 -0
- package/lib/typescript/auth/index.d.ts +2 -0
- package/lib/typescript/auth/index.d.ts.map +1 -0
- package/lib/typescript/components/AuthenticatorVerify.d.ts +3 -0
- package/lib/typescript/components/AuthenticatorVerify.d.ts.map +1 -0
- package/lib/typescript/components/FirebaseSignIn.d.ts +6 -0
- package/lib/typescript/components/FirebaseSignIn.d.ts.map +1 -0
- package/lib/typescript/components/PhoneVerify.d.ts +6 -0
- package/lib/typescript/components/PhoneVerify.d.ts.map +1 -0
- package/lib/typescript/components/TwoFactorAuthModal.d.ts +3 -0
- package/lib/typescript/components/TwoFactorAuthModal.d.ts.map +1 -0
- package/lib/typescript/components/index.d.ts +3 -0
- package/lib/typescript/components/index.d.ts.map +1 -0
- package/lib/typescript/contexts/FirebaseContext.d.ts +9 -0
- package/lib/typescript/contexts/FirebaseContext.d.ts.map +1 -0
- package/lib/typescript/contexts/index.d.ts +2 -0
- package/lib/typescript/contexts/index.d.ts.map +1 -0
- package/lib/typescript/db/index.d.ts +2 -0
- package/lib/typescript/db/index.d.ts.map +1 -0
- package/lib/typescript/db/utils.d.ts +6 -0
- package/lib/typescript/db/utils.d.ts.map +1 -0
- package/lib/typescript/hooks/backend.d.ts +2 -0
- package/lib/typescript/hooks/backend.d.ts.map +1 -0
- package/lib/typescript/hooks/index.d.ts +2 -0
- package/lib/typescript/hooks/index.d.ts.map +1 -0
- package/lib/typescript/hooks/useAuthenticatorVerify.d.ts +8 -0
- package/lib/typescript/hooks/useAuthenticatorVerify.d.ts.map +1 -0
- package/lib/typescript/hooks/usePhoneVerify.d.ts +9 -0
- package/lib/typescript/hooks/usePhoneVerify.d.ts.map +1 -0
- package/lib/typescript/icons/Google.d.ts +5 -0
- package/lib/typescript/icons/Google.d.ts.map +1 -0
- package/lib/typescript/index.d.ts +10 -0
- package/lib/typescript/index.d.ts.map +1 -0
- package/lib/typescript/storage/index.d.ts +2 -0
- package/lib/typescript/storage/index.d.ts.map +1 -0
- package/lib/typescript/storage/utils.d.ts +4 -0
- package/lib/typescript/storage/utils.d.ts.map +1 -0
- package/lib/typescript/types/adapter.d.ts +6 -0
- package/lib/typescript/types/adapter.d.ts.map +1 -0
- package/lib/typescript/types/auth.d.ts +12 -0
- package/lib/typescript/types/auth.d.ts.map +1 -0
- package/lib/typescript/types/db.d.ts +8 -0
- package/lib/typescript/types/db.d.ts.map +1 -0
- package/lib/typescript/types/functions.d.ts +3 -0
- package/lib/typescript/types/functions.d.ts.map +1 -0
- package/lib/typescript/types/index.d.ts +24 -0
- package/lib/typescript/types/index.d.ts.map +1 -0
- package/lib/typescript/types/storage.d.ts +3 -0
- package/lib/typescript/types/storage.d.ts.map +1 -0
- package/lib/typescript/utils/validation.d.ts +21 -0
- package/lib/typescript/utils/validation.d.ts.map +1 -0
- package/package.json +29 -12
- package/src/adapter/auth.ts +474 -0
- package/src/adapter/db.ts +146 -0
- package/src/adapter/index.ts +30 -0
- package/src/adapter/storage.ts +58 -0
- package/src/auth/functions.ts +7 -0
- package/src/auth/index.ts +1 -0
- package/src/components/AuthenticatorVerify.tsx +75 -0
- package/src/components/FirebaseSignIn.tsx +187 -0
- package/src/components/PhoneVerify.tsx +102 -0
- package/src/components/TwoFactorAuthModal.tsx +133 -0
- package/src/components/index.ts +2 -0
- package/src/contexts/FirebaseContext.tsx +54 -0
- package/src/contexts/index.ts +1 -0
- package/src/db/index.ts +1 -0
- package/src/db/utils.ts +142 -0
- package/src/hooks/backend.ts +4 -0
- package/src/hooks/index.ts +1 -0
- package/src/hooks/useAuthenticatorVerify.ts +45 -0
- package/src/hooks/usePhoneVerify.ts +76 -0
- package/src/icons/Google.tsx +24 -0
- package/src/index.ts +9 -0
- package/src/storage/index.ts +1 -0
- package/src/storage/utils.ts +29 -0
- package/src/types/adapter.ts +13 -0
- package/src/types/auth.ts +13 -0
- package/src/types/db.ts +10 -0
- package/src/types/functions.ts +3 -0
- package/src/types/index.ts +26 -0
- package/src/types/storage.ts +3 -0
- 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
|
+
})
|