2025-06-24 17:17:15 +01:00
|
|
|
import 'package:cake_wallet/core/secure_storage.dart';
|
|
|
|
import 'package:cake_wallet/entities/secret_store_key.dart';
|
|
|
|
import 'package:cake_wallet/store/authentication_store.dart';
|
|
|
|
import 'package:cake_wallet/store/settings_store.dart';
|
|
|
|
import 'package:cake_wallet/entities/preferences_key.dart';
|
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
import 'package:cw_core/utils/print_verbose.dart';
|
|
|
|
|
|
|
|
class ResetService {
|
|
|
|
ResetService({
|
|
|
|
required this.secureStorage,
|
|
|
|
required this.authenticationStore,
|
|
|
|
required this.settingsStore,
|
|
|
|
});
|
|
|
|
|
|
|
|
final SecureStorage secureStorage;
|
|
|
|
final AuthenticationStore authenticationStore;
|
|
|
|
final SettingsStore settingsStore;
|
|
|
|
|
|
|
|
static const List<String> _authKeys = [
|
|
|
|
SecureKey.allowBiometricalAuthenticationKey,
|
|
|
|
SecureKey.useTOTP2FA,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForAccessingWallet,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForSendsToContact,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForSendsToNonContact,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForSendsToInternalWallets,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForExchangesToInternalWallets,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForExchangesToExternalWallets,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForAddingContacts,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForCreatingNewWallets,
|
|
|
|
SecureKey.shouldRequireTOTP2FAForAllSecurityAndBackupSettings,
|
|
|
|
SecureKey.selectedCake2FAPreset,
|
|
|
|
SecureKey.totpSecretKey,
|
|
|
|
SecureKey.pinTimeOutDuration,
|
|
|
|
SecureKey.lastAuthTimeMilliseconds,
|
|
|
|
'PIN_CODE_PASSWORD',
|
|
|
|
];
|
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
static const List<String> _walletPreferencesKeys = [
|
|
|
|
PreferencesKey.currentWalletName,
|
|
|
|
PreferencesKey.currentWalletType,
|
|
|
|
];
|
|
|
|
|
2025-06-24 17:17:15 +01:00
|
|
|
bool _isAuthKey(String key) => _authKeys.contains(key);
|
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
/// Checks if this is a new install and clears any existing auth data from Storage
|
2025-06-24 17:17:15 +01:00
|
|
|
Future<void> resetAuthDataOnNewInstall(SharedPreferences sharedPreferences) async {
|
|
|
|
try {
|
|
|
|
final isNewInstall = sharedPreferences.getBool(PreferencesKey.isNewInstall) ?? false;
|
|
|
|
|
|
|
|
if (isNewInstall) {
|
2025-06-25 06:02:07 +01:00
|
|
|
await _clearExistingAuthDataOnNewInstall(sharedPreferences);
|
2025-06-24 17:17:15 +01:00
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
printV('Error during new install auth reset: $e');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks if there's existing auth data that should be cleared on new install
|
2025-06-25 06:02:07 +01:00
|
|
|
Future<void> _clearExistingAuthDataOnNewInstall(SharedPreferences sharedPreferences) async {
|
2025-06-24 17:17:15 +01:00
|
|
|
final allKeys = await secureStorage.readAll();
|
|
|
|
final authKeysFound = <String>[];
|
2025-06-25 06:02:07 +01:00
|
|
|
final walletPrefsFound = <String>[];
|
2025-06-24 17:17:15 +01:00
|
|
|
|
|
|
|
for (final key in allKeys.keys) {
|
|
|
|
if (_isAuthKey(key)) {
|
|
|
|
authKeysFound.add(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
for (final key in _walletPreferencesKeys) {
|
|
|
|
if (sharedPreferences.containsKey(key)) {
|
|
|
|
walletPrefsFound.add(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (authKeysFound.isNotEmpty || walletPrefsFound.isNotEmpty) {
|
2025-06-24 17:17:15 +01:00
|
|
|
printV(
|
|
|
|
'Found ${authKeysFound.length} existing auth keys in storage: ${authKeysFound.join(', ')}',
|
|
|
|
);
|
2025-06-25 06:02:07 +01:00
|
|
|
printV(
|
|
|
|
'Found ${walletPrefsFound.length} existing wallet preferences: ${walletPrefsFound.join(', ')}',
|
|
|
|
);
|
2025-06-24 17:17:15 +01:00
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
await resetAuthenticationData(sharedPreferences);
|
2025-06-24 17:17:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
/// Resets authentication data auth store, storage and settings store
|
|
|
|
Future<void> resetAuthenticationData(SharedPreferences sharedPreferences) async {
|
2025-06-24 17:17:15 +01:00
|
|
|
try {
|
2025-06-25 06:02:07 +01:00
|
|
|
authenticationStore.state = AuthenticationState.uninitialized;
|
|
|
|
|
2025-06-24 17:17:15 +01:00
|
|
|
await Future.wait([
|
|
|
|
_deleteAuthenticationKeys(),
|
|
|
|
_resetSettingsStoreAuthData(),
|
2025-06-25 06:02:07 +01:00
|
|
|
_clearWalletPreferences(sharedPreferences),
|
2025-06-24 17:17:15 +01:00
|
|
|
]);
|
|
|
|
} catch (e) {
|
|
|
|
printV('An error occurred during authentication reset: $e');
|
|
|
|
rethrow;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Resets authentication-related data in SettingsStore to default values
|
|
|
|
Future<void> _resetSettingsStoreAuthData() async {
|
|
|
|
settingsStore.useTOTP2FA = false;
|
|
|
|
settingsStore.totpSecretKey = '';
|
|
|
|
settingsStore.shouldRequireTOTP2FAForAccessingWallet = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForSendsToContact = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForSendsToNonContact = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForSendsToInternalWallets = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForExchangesToInternalWallets = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForExchangesToExternalWallets = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForAddingContacts = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForCreatingNewWallets = false;
|
|
|
|
settingsStore.shouldRequireTOTP2FAForAllSecurityAndBackupSettings = false;
|
|
|
|
settingsStore.allowBiometricalAuthentication = false;
|
|
|
|
}
|
|
|
|
|
2025-06-25 06:02:07 +01:00
|
|
|
Future<void> _clearWalletPreferences(SharedPreferences sharedPreferences) async {
|
|
|
|
final failedDeletions = <String>[];
|
|
|
|
|
|
|
|
for (final key in _walletPreferencesKeys) {
|
|
|
|
try {
|
|
|
|
await sharedPreferences.remove(key);
|
|
|
|
} catch (e) {
|
|
|
|
failedDeletions.add(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (failedDeletions.isNotEmpty) {
|
|
|
|
printV(
|
|
|
|
'Warning: Failed to delete ${failedDeletions.length} wallet preferences: ${failedDeletions.join(', ')}',
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
printV('All wallet preferences deleted successfully');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-06-24 17:17:15 +01:00
|
|
|
Future<void> _deleteAuthenticationKeys() async {
|
|
|
|
final failedDeletions = <String>[];
|
|
|
|
|
|
|
|
final deletionFutures = _authKeys.map((key) async {
|
|
|
|
try {
|
|
|
|
await secureStorage.delete(key: key);
|
|
|
|
} catch (e) {
|
|
|
|
failedDeletions.add(key);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
await Future.wait(deletionFutures);
|
|
|
|
|
|
|
|
if (failedDeletions.isNotEmpty) {
|
|
|
|
printV(
|
|
|
|
'Warning: Failed to delete ${failedDeletions.length} auth keys: ${failedDeletions.join(', ')}',
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
printV('All auth keys deleted successfully');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|