@onairos/react-native 3.1.16 → 3.1.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 +404 -0
- package/lib/commonjs/assets/images/Checkbox.svg +3 -3
- package/lib/commonjs/assets/images/EnochE.svg +19 -19
- package/lib/commonjs/assets/images/Personalityprofile.svg +3 -3
- package/lib/commonjs/assets/images/Personalitytraits.svg +3 -3
- package/lib/commonjs/assets/images/Userpreferences.svg +3 -3
- package/lib/commonjs/assets/images/arrow.svg +20 -20
- package/lib/commonjs/assets/images/basicproficon.svg +43 -43
- package/lib/commonjs/assets/images/basicprofile.svg +3 -3
- package/lib/commonjs/assets/images/checkmark.svg +4 -4
- package/lib/commonjs/assets/images/contentanalysis.svg +3 -3
- package/lib/commonjs/assets/images/contenticon.svg +23 -23
- package/lib/commonjs/assets/images/personalityicon.svg +18 -18
- package/lib/commonjs/assets/images/x-close.svg +3 -3
- package/lib/commonjs/components/ModalSheet.js +8 -2
- package/lib/commonjs/components/ModalSheet.js.map +1 -1
- package/lib/commonjs/components/OnairosButton.js +290 -0
- package/lib/commonjs/components/OnairosButton.js.map +1 -0
- package/lib/commonjs/components/OnairosSignInButton.js +32 -8
- package/lib/commonjs/components/OnairosSignInButton.js.map +1 -1
- package/lib/commonjs/components/UniversalOnboarding.js +4 -4
- package/lib/commonjs/components/WelcomeScreen.js +29 -13
- package/lib/commonjs/components/WelcomeScreen.js.map +1 -1
- package/lib/commonjs/config/api.js +2 -2
- package/lib/commonjs/hooks/useConnections.js +6 -6
- package/lib/commonjs/hooks/useUserConnections.js +10 -10
- package/lib/commonjs/index.js +13 -6
- package/lib/commonjs/index.js.map +1 -1
- package/lib/commonjs/services/apiClient.js +35 -35
- package/lib/commonjs/services/apiKeyService.js +99 -99
- package/lib/commonjs/services/authService.js +82 -82
- package/lib/commonjs/services/biometricPinService.js +10 -10
- package/lib/commonjs/services/connectedAccountsService.js +32 -32
- package/lib/commonjs/services/googleAuthService.js +15 -15
- package/lib/commonjs/services/imageCompressionService.js +15 -15
- package/lib/commonjs/services/jwtStorageService.js +59 -59
- package/lib/commonjs/services/mobileTrainingService.js +14 -14
- package/lib/commonjs/services/pinEncryptionService.js +10 -10
- package/lib/commonjs/services/pinStorageUtils.js +15 -15
- package/lib/commonjs/services/platformAuthService.js +47 -47
- package/lib/commonjs/services/storageService.js +31 -31
- package/lib/commonjs/services/trainingApiHelpers.js +33 -33
- package/lib/commonjs/services/userConnectionsService.js +24 -24
- package/lib/commonjs/utils/Portal.js +4 -4
- package/lib/commonjs/utils/api.js +24 -24
- package/lib/commonjs/utils/auth.js +18 -18
- package/lib/commonjs/utils/crypto.js +13 -13
- package/lib/commonjs/utils/encryption.js +12 -12
- package/lib/commonjs/utils/eventUtils.js +52 -52
- package/lib/commonjs/utils/programmaticFlow.js +16 -16
- package/lib/commonjs/utils/retryHelper.js +27 -27
- package/lib/module/assets/images/Checkbox.svg +3 -3
- package/lib/module/assets/images/EnochE.svg +19 -19
- package/lib/module/assets/images/Personalityprofile.svg +3 -3
- package/lib/module/assets/images/Personalitytraits.svg +3 -3
- package/lib/module/assets/images/Userpreferences.svg +3 -3
- package/lib/module/assets/images/arrow.svg +20 -20
- package/lib/module/assets/images/basicproficon.svg +43 -43
- package/lib/module/assets/images/basicprofile.svg +3 -3
- package/lib/module/assets/images/checkmark.svg +4 -4
- package/lib/module/assets/images/contentanalysis.svg +3 -3
- package/lib/module/assets/images/contenticon.svg +23 -23
- package/lib/module/assets/images/personalityicon.svg +18 -18
- package/lib/module/assets/images/x-close.svg +3 -3
- package/lib/module/components/ModalSheet.js +7 -2
- package/lib/module/components/ModalSheet.js.map +1 -1
- package/lib/module/components/OnairosButton.js +282 -0
- package/lib/module/components/OnairosButton.js.map +1 -0
- package/lib/module/components/OnairosSignInButton.js +32 -8
- package/lib/module/components/OnairosSignInButton.js.map +1 -1
- package/lib/module/components/UniversalOnboarding.js +4 -4
- package/lib/module/components/WelcomeScreen.js +25 -10
- package/lib/module/components/WelcomeScreen.js.map +1 -1
- package/lib/module/config/api.js +2 -2
- package/lib/module/hooks/useConnections.js +6 -6
- package/lib/module/hooks/useUserConnections.js +10 -10
- package/lib/module/index.js +11 -11
- package/lib/module/index.js.map +1 -1
- package/lib/module/services/apiClient.js +35 -35
- package/lib/module/services/apiKeyService.js +99 -99
- package/lib/module/services/authService.js +82 -82
- package/lib/module/services/biometricPinService.js +10 -10
- package/lib/module/services/connectedAccountsService.js +32 -32
- package/lib/module/services/googleAuthService.js +15 -15
- package/lib/module/services/imageCompressionService.js +15 -15
- package/lib/module/services/jwtStorageService.js +59 -59
- package/lib/module/services/mobileTrainingService.js +14 -14
- package/lib/module/services/pinEncryptionService.js +10 -10
- package/lib/module/services/pinStorageUtils.js +15 -15
- package/lib/module/services/platformAuthService.js +47 -47
- package/lib/module/services/storageService.js +31 -31
- package/lib/module/services/trainingApiHelpers.js +33 -33
- package/lib/module/services/userConnectionsService.js +24 -24
- package/lib/module/utils/Portal.js +4 -4
- package/lib/module/utils/api.js +24 -24
- package/lib/module/utils/auth.js +18 -18
- package/lib/module/utils/crypto.js +13 -13
- package/lib/module/utils/encryption.js +12 -12
- package/lib/module/utils/eventUtils.js +52 -52
- package/lib/module/utils/programmaticFlow.js +16 -16
- package/lib/module/utils/retryHelper.js +27 -27
- package/lib/typescript/components/ModalSheet.d.ts.map +1 -1
- package/lib/typescript/components/OnairosButton.d.ts +37 -0
- package/lib/typescript/components/OnairosButton.d.ts.map +1 -0
- package/lib/typescript/components/OnairosSignInButton.d.ts.map +1 -1
- package/lib/typescript/components/WelcomeScreen.d.ts.map +1 -1
- package/lib/typescript/index.d.ts +3 -1
- package/lib/typescript/index.d.ts.map +1 -1
- package/package.json +145 -163
- package/src/api/index.ts +151 -151
- package/src/assets/images/Checkbox.svg +3 -3
- package/src/assets/images/EnochE.svg +19 -19
- package/src/assets/images/Personalityprofile.svg +3 -3
- package/src/assets/images/Personalitytraits.svg +3 -3
- package/src/assets/images/Userpreferences.svg +3 -3
- package/src/assets/images/arrow.svg +20 -20
- package/src/assets/images/basicproficon.svg +43 -43
- package/src/assets/images/basicprofile.svg +3 -3
- package/src/assets/images/checkmark.svg +4 -4
- package/src/assets/images/contentanalysis.svg +3 -3
- package/src/assets/images/contenticon.svg +23 -23
- package/src/assets/images/personalityicon.svg +18 -18
- package/src/assets/images/x-close.svg +3 -3
- package/src/components/BodyText.tsx +33 -33
- package/src/components/BrandMark.tsx +62 -62
- package/src/components/CodeInput.tsx +32 -32
- package/src/components/DataRequestScreen.tsx +355 -355
- package/src/components/EmailInput.tsx +31 -31
- package/src/components/EmailVerificationModal.tsx +363 -363
- package/src/components/ExistingUserDataConfirmation.tsx +506 -506
- package/src/components/GoogleButton.tsx +55 -55
- package/src/components/HeadingGroup.tsx +49 -49
- package/src/components/ModalHeader.tsx +125 -125
- package/src/components/ModalSheet.tsx +59 -57
- package/src/components/Onairos.tsx +422 -422
- package/src/components/OnairosButton.tsx +339 -0
- package/src/components/OnairosSignInButton.tsx +31 -9
- package/src/components/Overlay.tsx +506 -506
- package/src/components/PersonaImage.tsx +79 -79
- package/src/components/PersonaLoadingScreen.tsx +201 -201
- package/src/components/PersonalizationConsentScreen.tsx +410 -410
- package/src/components/PinCreationScreen.tsx +492 -492
- package/src/components/PinInput.tsx +555 -555
- package/src/components/PlatformConnectorsStep.tsx +891 -891
- package/src/components/PlatformList.tsx +144 -144
- package/src/components/PlatformToggle.tsx +226 -226
- package/src/components/PrimaryButton.tsx +213 -213
- package/src/components/SignInMatchAnimation.tsx +225 -225
- package/src/components/SignInStep.tsx +217 -217
- package/src/components/TrainingModal.tsx +1047 -1047
- package/src/components/UniversalOnboarding.tsx +2887 -2887
- package/src/components/VerificationStep.tsx +198 -198
- package/src/components/WelcomeScreen.tsx +490 -473
- package/src/components/icons/Basicproficon.tsx +30 -30
- package/src/components/icons/Basicprofile.tsx +17 -17
- package/src/components/icons/Checkbox.tsx +17 -17
- package/src/components/icons/Checkmark.tsx +24 -24
- package/src/components/icons/Contentanalysis.tsx +17 -17
- package/src/components/icons/Contenticon.tsx +30 -30
- package/src/components/icons/EnochE.tsx +39 -39
- package/src/components/icons/Personalityicon.tsx +22 -22
- package/src/components/icons/Personalityprofile.tsx +17 -17
- package/src/components/icons/Personalitytraits.tsx +17 -17
- package/src/components/icons/Userpreferences.tsx +17 -17
- package/src/components/icons/index.ts +12 -12
- package/src/components/onboarding/OAuthWebView.tsx +232 -232
- package/src/config/api.ts +25 -25
- package/src/context/AuthContext.tsx +393 -393
- package/src/hooks/useConnectedAccounts.ts +138 -138
- package/src/hooks/useConnections.ts +161 -161
- package/src/hooks/useCredentials.ts +174 -174
- package/src/hooks/useUserConnections.ts +165 -165
- package/src/index.js +14 -0
- package/src/index.ts +99 -96
- package/src/services/apiClient.ts +336 -336
- package/src/services/apiKeyService.ts +919 -919
- package/src/services/authService.ts +1008 -1008
- package/src/services/biometricPinService.ts +192 -192
- package/src/services/connectedAccountsService.ts +289 -289
- package/src/services/googleAuthService.ts +279 -279
- package/src/services/imageCompressionService.ts +302 -302
- package/src/services/jwtStorageService.ts +256 -256
- package/src/services/mobileTrainingService.ts +203 -203
- package/src/services/pinEncryptionService.ts +75 -75
- package/src/services/pinStorageUtils.ts +96 -96
- package/src/services/platformAuthService.ts +1346 -1346
- package/src/services/storageService.ts +451 -451
- package/src/services/trainingApiHelpers.ts +66 -66
- package/src/services/userConnectionsService.ts +556 -556
- package/src/services/youtubeMigrationService.ts +453 -453
- package/src/theme/index.ts +239 -239
- package/src/types/ambient.d.ts +28 -28
- package/src/types/index.ts +265 -265
- package/src/types/node-fix.d.ts +18 -18
- package/src/types/node-override.d.ts +23 -23
- package/src/types/opacity.d.ts +15 -15
- package/src/types/types.d.ts +17 -17
- package/src/utils/Portal.tsx +82 -82
- package/src/utils/api.js +111 -111
- package/src/utils/auth.js +103 -103
- package/src/utils/crypto.js +59 -59
- package/src/utils/encryption.ts +68 -68
- package/src/utils/eventUtils.ts +302 -302
- package/src/utils/haptics.ts +58 -58
- package/src/utils/imagePreloader.ts +2 -2
- package/src/utils/programmaticFlow.ts +112 -112
- package/src/utils/retryHelper.ts +274 -274
|
@@ -1,193 +1,193 @@
|
|
|
1
|
-
import * as Keychain from 'react-native-keychain';
|
|
2
|
-
import { Platform } from 'react-native';
|
|
3
|
-
|
|
4
|
-
// Service key for PIN storage
|
|
5
|
-
const PIN_SERVICE_KEY = 'OnairosEventsPIN';
|
|
6
|
-
const PIN_USERNAME = 'user_pin';
|
|
7
|
-
|
|
8
|
-
export interface BiometricPinService {
|
|
9
|
-
storePinWithBiometric: (pin: string) => Promise<boolean>;
|
|
10
|
-
retrievePinWithBiometric: () => Promise<string | null>;
|
|
11
|
-
isPinStored: () => Promise<boolean>;
|
|
12
|
-
removePinFromStorage: () => Promise<boolean>;
|
|
13
|
-
isBiometricAvailable: () => Promise<boolean>;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Check if biometric authentication is available on the device
|
|
18
|
-
*/
|
|
19
|
-
export const isBiometricAvailable = async (): Promise<boolean> => {
|
|
20
|
-
try {
|
|
21
|
-
const biometryType = await Keychain.getSupportedBiometryType();
|
|
22
|
-
console.log('📱 Biometric support:', biometryType);
|
|
23
|
-
|
|
24
|
-
// Check for Face ID on iOS or any biometric on Android
|
|
25
|
-
return biometryType === Keychain.BIOMETRY_TYPE.FACE_ID ||
|
|
26
|
-
biometryType === Keychain.BIOMETRY_TYPE.TOUCH_ID ||
|
|
27
|
-
biometryType === Keychain.BIOMETRY_TYPE.FINGERPRINT ||
|
|
28
|
-
biometryType === Keychain.BIOMETRY_TYPE.FACE ||
|
|
29
|
-
biometryType === Keychain.BIOMETRY_TYPE.IRIS;
|
|
30
|
-
} catch (error) {
|
|
31
|
-
console.error('❌ Error checking biometric availability:', error);
|
|
32
|
-
return false;
|
|
33
|
-
}
|
|
34
|
-
};
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Store PIN with biometric authentication
|
|
38
|
-
*/
|
|
39
|
-
export const storePinWithBiometric = async (pin: string): Promise<boolean> => {
|
|
40
|
-
try {
|
|
41
|
-
console.log('🔐 Starting Face ID authentication for PIN storage...');
|
|
42
|
-
|
|
43
|
-
// Check if biometric is available - REQUIRED for this implementation
|
|
44
|
-
const biometricAvailable = await isBiometricAvailable();
|
|
45
|
-
if (!biometricAvailable) {
|
|
46
|
-
console.error('❌ Face ID/Touch ID not available on this device');
|
|
47
|
-
return false;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
console.log('📱 Face ID available, showing authentication prompt...');
|
|
51
|
-
|
|
52
|
-
// Configure keychain options for MANDATORY biometric authentication
|
|
53
|
-
const keychainOptions = {
|
|
54
|
-
accessControl: Keychain.ACCESS_CONTROL.BIOMETRY_CURRENT_SET,
|
|
55
|
-
authenticationType: Keychain.AUTHENTICATION_TYPE.BIOMETRICS,
|
|
56
|
-
touchID: true,
|
|
57
|
-
showModal: true,
|
|
58
|
-
kLocalizedFallbackTitle: '', // Disable fallback to prevent passcode option
|
|
59
|
-
// Add explicit prompts for better UX
|
|
60
|
-
...(Platform.OS === 'ios' && {
|
|
61
|
-
localizedPrompt: 'Authenticate with Face ID to secure your PIN',
|
|
62
|
-
localizedCancel: 'Cancel',
|
|
63
|
-
}),
|
|
64
|
-
...(Platform.OS === 'android' && {
|
|
65
|
-
promptMessage: 'Authenticate with biometric to secure your PIN',
|
|
66
|
-
cancelButtonText: 'Cancel',
|
|
67
|
-
})
|
|
68
|
-
} as const;
|
|
69
|
-
|
|
70
|
-
// Store the PIN - this will trigger the Face ID prompt
|
|
71
|
-
console.log('👤 Requesting Face ID authentication...');
|
|
72
|
-
const result = await Keychain.setInternetCredentials(
|
|
73
|
-
PIN_SERVICE_KEY,
|
|
74
|
-
PIN_USERNAME,
|
|
75
|
-
pin,
|
|
76
|
-
keychainOptions
|
|
77
|
-
);
|
|
78
|
-
|
|
79
|
-
console.log('✅ Face ID authentication successful - PIN stored securely');
|
|
80
|
-
console.log('🔍 Keychain storage result:', result);
|
|
81
|
-
return true;
|
|
82
|
-
|
|
83
|
-
} catch (error: any) {
|
|
84
|
-
console.error('❌ Face ID authentication failed:', error);
|
|
85
|
-
console.error('❌ Error details:', {
|
|
86
|
-
message: error.message,
|
|
87
|
-
code: error.code,
|
|
88
|
-
domain: error.domain,
|
|
89
|
-
userInfo: error.userInfo
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
// Handle specific error cases
|
|
93
|
-
if (error?.message?.includes('UserCancel') || error?.code === -128) {
|
|
94
|
-
console.log('👤 User cancelled Face ID authentication');
|
|
95
|
-
return false;
|
|
96
|
-
} else if (error?.message?.includes('BiometryNotAvailable') || error?.code === -6) {
|
|
97
|
-
console.log('📱 Face ID/Touch ID not available');
|
|
98
|
-
return false;
|
|
99
|
-
} else if (error?.message?.includes('AuthenticationFailed') || error?.code === -1) {
|
|
100
|
-
console.log('🚫 Face ID authentication failed');
|
|
101
|
-
return false;
|
|
102
|
-
} else if (error?.message?.includes('BiometryLockout') || error?.code === -8) {
|
|
103
|
-
console.log('🔒 Face ID locked out - too many failed attempts');
|
|
104
|
-
return false;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
console.error('❌ Unknown Face ID error:', error.message);
|
|
108
|
-
return false;
|
|
109
|
-
}
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
/**
|
|
113
|
-
* Retrieve PIN with biometric authentication
|
|
114
|
-
*/
|
|
115
|
-
export const retrievePinWithBiometric = async (): Promise<string | null> => {
|
|
116
|
-
try {
|
|
117
|
-
console.log('🔓 Retrieving PIN with biometric authentication...');
|
|
118
|
-
|
|
119
|
-
// Retrieve the PIN (will prompt for biometric if configured)
|
|
120
|
-
const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
|
|
121
|
-
|
|
122
|
-
if (credentials && credentials.password) {
|
|
123
|
-
console.log('✅ PIN retrieved successfully');
|
|
124
|
-
return credentials.password;
|
|
125
|
-
} else {
|
|
126
|
-
console.log('⚠️ No PIN found in secure storage');
|
|
127
|
-
return null;
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
} catch (error: any) {
|
|
131
|
-
console.error('❌ Error retrieving PIN with biometric:', error);
|
|
132
|
-
|
|
133
|
-
// Handle specific error cases
|
|
134
|
-
if (error?.message?.includes('UserCancel')) {
|
|
135
|
-
console.log('👤 User cancelled biometric authentication');
|
|
136
|
-
return null;
|
|
137
|
-
} else if (error?.message?.includes('BiometryNotAvailable')) {
|
|
138
|
-
console.log('📱 Biometric not available');
|
|
139
|
-
return null;
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
return null;
|
|
143
|
-
}
|
|
144
|
-
};
|
|
145
|
-
|
|
146
|
-
/**
|
|
147
|
-
* Check if PIN is stored in secure storage
|
|
148
|
-
*/
|
|
149
|
-
export const isPinStored = async (): Promise<boolean> => {
|
|
150
|
-
try {
|
|
151
|
-
const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
|
|
152
|
-
return !!(credentials && credentials.password);
|
|
153
|
-
} catch (error: any) {
|
|
154
|
-
console.error('❌ Error checking PIN storage:', error);
|
|
155
|
-
return false;
|
|
156
|
-
}
|
|
157
|
-
};
|
|
158
|
-
|
|
159
|
-
/**
|
|
160
|
-
* Remove PIN from secure storage
|
|
161
|
-
*/
|
|
162
|
-
export const removePinFromStorage = async (): Promise<boolean> => {
|
|
163
|
-
try {
|
|
164
|
-
console.log('🗑️ Removing PIN from secure storage...');
|
|
165
|
-
|
|
166
|
-
// For v10.0.0, we need to use resetGenericPassword instead of resetInternetCredentials
|
|
167
|
-
// or handle the API difference properly
|
|
168
|
-
try {
|
|
169
|
-
await Keychain.resetGenericPassword({ service: PIN_SERVICE_KEY });
|
|
170
|
-
console.log('✅ PIN removed successfully');
|
|
171
|
-
return true;
|
|
172
|
-
} catch (resetError) {
|
|
173
|
-
// Fallback: try the old API
|
|
174
|
-
console.log('🔄 Trying alternative reset method...');
|
|
175
|
-
const result = await (Keychain as any).resetInternetCredentials(PIN_SERVICE_KEY);
|
|
176
|
-
console.log('✅ PIN removed successfully with fallback method');
|
|
177
|
-
return true;
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
} catch (error: any) {
|
|
181
|
-
console.error('❌ Error removing PIN from storage:', error);
|
|
182
|
-
return false;
|
|
183
|
-
}
|
|
184
|
-
};
|
|
185
|
-
|
|
186
|
-
// Export the service object
|
|
187
|
-
export const biometricPinService: BiometricPinService = {
|
|
188
|
-
storePinWithBiometric,
|
|
189
|
-
retrievePinWithBiometric,
|
|
190
|
-
isPinStored,
|
|
191
|
-
removePinFromStorage,
|
|
192
|
-
isBiometricAvailable,
|
|
1
|
+
import * as Keychain from 'react-native-keychain';
|
|
2
|
+
import { Platform } from 'react-native';
|
|
3
|
+
|
|
4
|
+
// Service key for PIN storage
|
|
5
|
+
const PIN_SERVICE_KEY = 'OnairosEventsPIN';
|
|
6
|
+
const PIN_USERNAME = 'user_pin';
|
|
7
|
+
|
|
8
|
+
export interface BiometricPinService {
|
|
9
|
+
storePinWithBiometric: (pin: string) => Promise<boolean>;
|
|
10
|
+
retrievePinWithBiometric: () => Promise<string | null>;
|
|
11
|
+
isPinStored: () => Promise<boolean>;
|
|
12
|
+
removePinFromStorage: () => Promise<boolean>;
|
|
13
|
+
isBiometricAvailable: () => Promise<boolean>;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Check if biometric authentication is available on the device
|
|
18
|
+
*/
|
|
19
|
+
export const isBiometricAvailable = async (): Promise<boolean> => {
|
|
20
|
+
try {
|
|
21
|
+
const biometryType = await Keychain.getSupportedBiometryType();
|
|
22
|
+
console.log('📱 Biometric support:', biometryType);
|
|
23
|
+
|
|
24
|
+
// Check for Face ID on iOS or any biometric on Android
|
|
25
|
+
return biometryType === Keychain.BIOMETRY_TYPE.FACE_ID ||
|
|
26
|
+
biometryType === Keychain.BIOMETRY_TYPE.TOUCH_ID ||
|
|
27
|
+
biometryType === Keychain.BIOMETRY_TYPE.FINGERPRINT ||
|
|
28
|
+
biometryType === Keychain.BIOMETRY_TYPE.FACE ||
|
|
29
|
+
biometryType === Keychain.BIOMETRY_TYPE.IRIS;
|
|
30
|
+
} catch (error) {
|
|
31
|
+
console.error('❌ Error checking biometric availability:', error);
|
|
32
|
+
return false;
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Store PIN with biometric authentication
|
|
38
|
+
*/
|
|
39
|
+
export const storePinWithBiometric = async (pin: string): Promise<boolean> => {
|
|
40
|
+
try {
|
|
41
|
+
console.log('🔐 Starting Face ID authentication for PIN storage...');
|
|
42
|
+
|
|
43
|
+
// Check if biometric is available - REQUIRED for this implementation
|
|
44
|
+
const biometricAvailable = await isBiometricAvailable();
|
|
45
|
+
if (!biometricAvailable) {
|
|
46
|
+
console.error('❌ Face ID/Touch ID not available on this device');
|
|
47
|
+
return false;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
console.log('📱 Face ID available, showing authentication prompt...');
|
|
51
|
+
|
|
52
|
+
// Configure keychain options for MANDATORY biometric authentication
|
|
53
|
+
const keychainOptions = {
|
|
54
|
+
accessControl: Keychain.ACCESS_CONTROL.BIOMETRY_CURRENT_SET,
|
|
55
|
+
authenticationType: Keychain.AUTHENTICATION_TYPE.BIOMETRICS,
|
|
56
|
+
touchID: true,
|
|
57
|
+
showModal: true,
|
|
58
|
+
kLocalizedFallbackTitle: '', // Disable fallback to prevent passcode option
|
|
59
|
+
// Add explicit prompts for better UX
|
|
60
|
+
...(Platform.OS === 'ios' && {
|
|
61
|
+
localizedPrompt: 'Authenticate with Face ID to secure your PIN',
|
|
62
|
+
localizedCancel: 'Cancel',
|
|
63
|
+
}),
|
|
64
|
+
...(Platform.OS === 'android' && {
|
|
65
|
+
promptMessage: 'Authenticate with biometric to secure your PIN',
|
|
66
|
+
cancelButtonText: 'Cancel',
|
|
67
|
+
})
|
|
68
|
+
} as const;
|
|
69
|
+
|
|
70
|
+
// Store the PIN - this will trigger the Face ID prompt
|
|
71
|
+
console.log('👤 Requesting Face ID authentication...');
|
|
72
|
+
const result = await Keychain.setInternetCredentials(
|
|
73
|
+
PIN_SERVICE_KEY,
|
|
74
|
+
PIN_USERNAME,
|
|
75
|
+
pin,
|
|
76
|
+
keychainOptions
|
|
77
|
+
);
|
|
78
|
+
|
|
79
|
+
console.log('✅ Face ID authentication successful - PIN stored securely');
|
|
80
|
+
console.log('🔍 Keychain storage result:', result);
|
|
81
|
+
return true;
|
|
82
|
+
|
|
83
|
+
} catch (error: any) {
|
|
84
|
+
console.error('❌ Face ID authentication failed:', error);
|
|
85
|
+
console.error('❌ Error details:', {
|
|
86
|
+
message: error.message,
|
|
87
|
+
code: error.code,
|
|
88
|
+
domain: error.domain,
|
|
89
|
+
userInfo: error.userInfo
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
// Handle specific error cases
|
|
93
|
+
if (error?.message?.includes('UserCancel') || error?.code === -128) {
|
|
94
|
+
console.log('👤 User cancelled Face ID authentication');
|
|
95
|
+
return false;
|
|
96
|
+
} else if (error?.message?.includes('BiometryNotAvailable') || error?.code === -6) {
|
|
97
|
+
console.log('📱 Face ID/Touch ID not available');
|
|
98
|
+
return false;
|
|
99
|
+
} else if (error?.message?.includes('AuthenticationFailed') || error?.code === -1) {
|
|
100
|
+
console.log('🚫 Face ID authentication failed');
|
|
101
|
+
return false;
|
|
102
|
+
} else if (error?.message?.includes('BiometryLockout') || error?.code === -8) {
|
|
103
|
+
console.log('🔒 Face ID locked out - too many failed attempts');
|
|
104
|
+
return false;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
console.error('❌ Unknown Face ID error:', error.message);
|
|
108
|
+
return false;
|
|
109
|
+
}
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Retrieve PIN with biometric authentication
|
|
114
|
+
*/
|
|
115
|
+
export const retrievePinWithBiometric = async (): Promise<string | null> => {
|
|
116
|
+
try {
|
|
117
|
+
console.log('🔓 Retrieving PIN with biometric authentication...');
|
|
118
|
+
|
|
119
|
+
// Retrieve the PIN (will prompt for biometric if configured)
|
|
120
|
+
const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
|
|
121
|
+
|
|
122
|
+
if (credentials && credentials.password) {
|
|
123
|
+
console.log('✅ PIN retrieved successfully');
|
|
124
|
+
return credentials.password;
|
|
125
|
+
} else {
|
|
126
|
+
console.log('⚠️ No PIN found in secure storage');
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
} catch (error: any) {
|
|
131
|
+
console.error('❌ Error retrieving PIN with biometric:', error);
|
|
132
|
+
|
|
133
|
+
// Handle specific error cases
|
|
134
|
+
if (error?.message?.includes('UserCancel')) {
|
|
135
|
+
console.log('👤 User cancelled biometric authentication');
|
|
136
|
+
return null;
|
|
137
|
+
} else if (error?.message?.includes('BiometryNotAvailable')) {
|
|
138
|
+
console.log('📱 Biometric not available');
|
|
139
|
+
return null;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
return null;
|
|
143
|
+
}
|
|
144
|
+
};
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* Check if PIN is stored in secure storage
|
|
148
|
+
*/
|
|
149
|
+
export const isPinStored = async (): Promise<boolean> => {
|
|
150
|
+
try {
|
|
151
|
+
const credentials = await Keychain.getInternetCredentials(PIN_SERVICE_KEY);
|
|
152
|
+
return !!(credentials && credentials.password);
|
|
153
|
+
} catch (error: any) {
|
|
154
|
+
console.error('❌ Error checking PIN storage:', error);
|
|
155
|
+
return false;
|
|
156
|
+
}
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* Remove PIN from secure storage
|
|
161
|
+
*/
|
|
162
|
+
export const removePinFromStorage = async (): Promise<boolean> => {
|
|
163
|
+
try {
|
|
164
|
+
console.log('🗑️ Removing PIN from secure storage...');
|
|
165
|
+
|
|
166
|
+
// For v10.0.0, we need to use resetGenericPassword instead of resetInternetCredentials
|
|
167
|
+
// or handle the API difference properly
|
|
168
|
+
try {
|
|
169
|
+
await Keychain.resetGenericPassword({ service: PIN_SERVICE_KEY });
|
|
170
|
+
console.log('✅ PIN removed successfully');
|
|
171
|
+
return true;
|
|
172
|
+
} catch (resetError) {
|
|
173
|
+
// Fallback: try the old API
|
|
174
|
+
console.log('🔄 Trying alternative reset method...');
|
|
175
|
+
const result = await (Keychain as any).resetInternetCredentials(PIN_SERVICE_KEY);
|
|
176
|
+
console.log('✅ PIN removed successfully with fallback method');
|
|
177
|
+
return true;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
} catch (error: any) {
|
|
181
|
+
console.error('❌ Error removing PIN from storage:', error);
|
|
182
|
+
return false;
|
|
183
|
+
}
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
// Export the service object
|
|
187
|
+
export const biometricPinService: BiometricPinService = {
|
|
188
|
+
storePinWithBiometric,
|
|
189
|
+
retrievePinWithBiometric,
|
|
190
|
+
isPinStored,
|
|
191
|
+
removePinFromStorage,
|
|
192
|
+
isBiometricAvailable,
|
|
193
193
|
};
|