CW-278 Enhance PIN timeout feature code (#886)

* CW-278 enhance pin timeout feature

* CW-278 enhance pin timeout feature

* Update flow to remove extension

* Replace pin request on other instances
This commit is contained in:
Godwin Asuquo 2023-04-20 03:54:25 +03:00 committed by GitHub
parent 27961f2f25
commit efef30f8eb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 102 additions and 111 deletions

View file

@ -1,10 +1,12 @@
import 'package:cake_wallet/routes.dart';
import 'package:cake_wallet/src/screens/auth/auth_page.dart';
import 'package:flutter/material.dart';
import 'package:mobx/mobx.dart'; import 'package:mobx/mobx.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:shared_preferences/shared_preferences.dart'; import 'package:shared_preferences/shared_preferences.dart';
import 'package:cake_wallet/entities/preferences_key.dart'; import 'package:cake_wallet/entities/preferences_key.dart';
import 'package:cake_wallet/entities/secret_store_key.dart'; import 'package:cake_wallet/entities/secret_store_key.dart';
import 'package:cake_wallet/entities/encrypt.dart'; import 'package:cake_wallet/entities/encrypt.dart';
import 'package:cake_wallet/di.dart';
import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/store/settings_store.dart';
class AuthService with Store { class AuthService with Store {
@ -14,6 +16,12 @@ class AuthService with Store {
required this.settingsStore, required this.settingsStore,
}); });
static const List<String> _alwaysAuthenticateRoutes = [
Routes.showKeys,
Routes.backup,
Routes.setupPin,
];
final FlutterSecureStorage secureStorage; final FlutterSecureStorage secureStorage;
final SharedPreferences sharedPreferences; final SharedPreferences sharedPreferences;
final SettingsStore settingsStore; final SettingsStore settingsStore;
@ -66,4 +74,33 @@ class AuthService with Store {
return timeDifference.inMinutes; return timeDifference.inMinutes;
} }
Future<void> authenticateAction(BuildContext context,
{Function(bool)? onAuthSuccess, String? route, Object? arguments}) async {
assert(route != null || onAuthSuccess != null,
'Either route or onAuthSuccess param must be passed.');
if (!requireAuth() && !_alwaysAuthenticateRoutes.contains(route)) {
if (onAuthSuccess != null) {
onAuthSuccess(true);
} else {
Navigator.of(context).pushNamed(
route ?? '',
arguments: arguments,
);
}
return;
}
Navigator.of(context).pushNamed(Routes.auth,
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) async {
if (!isAuthenticatedSuccessfully) {
onAuthSuccess?.call(false);
return;
}
if (onAuthSuccess != null) {
auth.close().then((value) => onAuthSuccess.call(true));
} else {
auth.close(route: route, arguments: arguments);
}
});
}
} }

View file

@ -499,7 +499,7 @@ Future setup(
} }
getIt.registerFactory(() => getIt.registerFactory(() =>
WalletListPage(walletListViewModel: getIt.get<WalletListViewModel>())); WalletListPage(walletListViewModel: getIt.get<WalletListViewModel>(), authService: getIt.get<AuthService>(),));
getIt.registerFactory(() { getIt.registerFactory(() {
final wallet = getIt.get<AppStore>().wallet!; final wallet = getIt.get<AppStore>().wallet!;
@ -593,7 +593,7 @@ Future setup(
getIt.registerFactory(() => ConnectionSyncPage(getIt.get<NodeListViewModel>(), getIt.get<DashboardViewModel>())); getIt.registerFactory(() => ConnectionSyncPage(getIt.get<NodeListViewModel>(), getIt.get<DashboardViewModel>()));
getIt.registerFactory(() => SecurityBackupPage(getIt.get<SecuritySettingsViewModel>())); getIt.registerFactory(() => SecurityBackupPage(getIt.get<SecuritySettingsViewModel>(), getIt.get<AuthService>()));
getIt.registerFactory(() => PrivacyPage(getIt.get<PrivacySettingsViewModel>())); getIt.registerFactory(() => PrivacyPage(getIt.get<PrivacySettingsViewModel>()));
@ -926,7 +926,7 @@ Future setup(
wallet: getIt.get<AppStore>().wallet!) wallet: getIt.get<AppStore>().wallet!)
); );
getIt.registerFactory(() => DesktopWalletSelectionDropDown(getIt.get<WalletListViewModel>())); getIt.registerFactory(() => DesktopWalletSelectionDropDown(getIt.get<WalletListViewModel>(), getIt.get<AuthService>()));
getIt.registerFactory(() => DesktopSidebarViewModel()); getIt.registerFactory(() => DesktopSidebarViewModel());

View file

@ -1,8 +1,9 @@
import 'package:another_flushbar/flushbar.dart'; import 'package:another_flushbar/flushbar.dart';
import 'package:cake_wallet/core/auth_service.dart';
import 'package:cake_wallet/di.dart';
import 'package:cake_wallet/entities/desktop_dropdown_item.dart'; import 'package:cake_wallet/entities/desktop_dropdown_item.dart';
import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/generated/i18n.dart';
import 'package:cake_wallet/routes.dart'; import 'package:cake_wallet/routes.dart';
import 'package:cake_wallet/src/screens/auth/auth_page.dart';
import 'package:cake_wallet/src/screens/dashboard/desktop_widgets/dropdown_item_widget.dart'; import 'package:cake_wallet/src/screens/dashboard/desktop_widgets/dropdown_item_widget.dart';
import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart'; import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart';
import 'package:cake_wallet/utils/show_bar.dart'; import 'package:cake_wallet/utils/show_bar.dart';
@ -16,8 +17,10 @@ import 'package:flutter_mobx/flutter_mobx.dart';
class DesktopWalletSelectionDropDown extends StatefulWidget { class DesktopWalletSelectionDropDown extends StatefulWidget {
final WalletListViewModel walletListViewModel; final WalletListViewModel walletListViewModel;
final AuthService _authService;
DesktopWalletSelectionDropDown(this.walletListViewModel, {Key? key}) : super(key: key); DesktopWalletSelectionDropDown(this.walletListViewModel, this._authService, {Key? key})
: super(key: key);
@override @override
State<DesktopWalletSelectionDropDown> createState() => _DesktopWalletSelectionDropDownState(); State<DesktopWalletSelectionDropDown> createState() => _DesktopWalletSelectionDropDownState();
@ -140,25 +143,12 @@ class _DesktopWalletSelectionDropDownState extends State<DesktopWalletSelectionD
} }
Future<void> _loadWallet(WalletListItem wallet) async { Future<void> _loadWallet(WalletListItem wallet) async {
if (await widget.walletListViewModel.checkIfAuthRequired()) { widget._authService.authenticateAction(context,
await Navigator.of(context).pushNamed(Routes.auth, onAuthSuccess: (isAuthenticatedSuccessfully) async {
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) async {
if (!isAuthenticatedSuccessfully) { if (!isAuthenticatedSuccessfully) {
return; return;
} }
try {
auth.changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name));
await widget.walletListViewModel.loadWallet(wallet);
auth.hideProgressText();
auth.close();
setState(() {});
} catch (e) {
auth.changeProcessText(
S.of(context).wallet_list_failed_to_load(wallet.name, e.toString()));
}
});
} else {
try { try {
changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name)); changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name));
await widget.walletListViewModel.loadWallet(wallet); await widget.walletListViewModel.loadWallet(wallet);
@ -167,7 +157,7 @@ class _DesktopWalletSelectionDropDownState extends State<DesktopWalletSelectionD
} catch (e) { } catch (e) {
changeProcessText(S.of(context).wallet_list_failed_to_load(wallet.name, e.toString())); changeProcessText(S.of(context).wallet_list_failed_to_load(wallet.name, e.toString()));
} }
} });
} }
void _navigateToCreateWallet() { void _navigateToCreateWallet() {

View file

@ -1,6 +1,6 @@
import 'package:cake_wallet/core/auth_service.dart';
import 'package:cake_wallet/entities/pin_code_required_duration.dart'; import 'package:cake_wallet/entities/pin_code_required_duration.dart';
import 'package:cake_wallet/routes.dart'; import 'package:cake_wallet/routes.dart';
import 'package:cake_wallet/src/screens/auth/auth_page.dart';
import 'package:cake_wallet/src/screens/base_page.dart'; import 'package:cake_wallet/src/screens/base_page.dart';
import 'package:cake_wallet/generated/i18n.dart'; import 'package:cake_wallet/generated/i18n.dart';
import 'package:cake_wallet/src/screens/pin_code/pin_code_widget.dart'; import 'package:cake_wallet/src/screens/pin_code/pin_code_widget.dart';
@ -13,7 +13,9 @@ import 'package:flutter/material.dart';
import 'package:flutter_mobx/flutter_mobx.dart'; import 'package:flutter_mobx/flutter_mobx.dart';
class SecurityBackupPage extends BasePage { class SecurityBackupPage extends BasePage {
SecurityBackupPage(this._securitySettingsViewModel); SecurityBackupPage(this._securitySettingsViewModel, this._authService);
final AuthService _authService;
@override @override
String get title => S.current.security_and_backup; String get title => S.current.security_and_backup;
@ -27,35 +29,24 @@ class SecurityBackupPage extends BasePage {
child: Column(mainAxisSize: MainAxisSize.min, children: [ child: Column(mainAxisSize: MainAxisSize.min, children: [
SettingsCellWithArrow( SettingsCellWithArrow(
title: S.current.show_keys, title: S.current.show_keys,
handler: (_) => Navigator.of(context).pushNamed(Routes.auth, handler: (_) => _authService.authenticateAction(context, route: Routes.showKeys),
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) {
if (isAuthenticatedSuccessfully) {
auth.close(route: Routes.showKeys);
}
}),
), ),
StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)), StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)),
SettingsCellWithArrow( SettingsCellWithArrow(
title: S.current.create_backup, title: S.current.create_backup,
handler: (_) => Navigator.of(context).pushNamed(Routes.auth, handler: (_) => _authService.authenticateAction(context, route: Routes.backup),
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) {
if (isAuthenticatedSuccessfully) {
auth.close(route: Routes.backup);
}
}),
), ),
StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)), StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)),
SettingsCellWithArrow( SettingsCellWithArrow(
title: S.current.settings_change_pin, title: S.current.settings_change_pin,
handler: (_) => Navigator.of(context).pushNamed(Routes.auth, handler: (_) => _authService.authenticateAction(
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) { context,
auth.close( route: Routes.setupPin,
route: isAuthenticatedSuccessfully ? Routes.setupPin : null,
arguments: (PinCodeState<PinCodeWidget> setupPinContext, String _) { arguments: (PinCodeState<PinCodeWidget> setupPinContext, String _) {
setupPinContext.close(); setupPinContext.close();
}, },
); ),
})), ),
StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)), StandardListSeparator(padding: EdgeInsets.symmetric(horizontal: 24)),
Observer(builder: (_) { Observer(builder: (_) {
return SettingsSwitcherCell( return SettingsSwitcherCell(
@ -63,8 +54,8 @@ class SecurityBackupPage extends BasePage {
value: _securitySettingsViewModel.allowBiometricalAuthentication, value: _securitySettingsViewModel.allowBiometricalAuthentication,
onValueChange: (BuildContext context, bool value) { onValueChange: (BuildContext context, bool value) {
if (value) { if (value) {
Navigator.of(context).pushNamed(Routes.auth, _authService.authenticateAction(context,
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) async { onAuthSuccess: (isAuthenticatedSuccessfully) async {
if (isAuthenticatedSuccessfully) { if (isAuthenticatedSuccessfully) {
if (await _securitySettingsViewModel.biometricAuthenticated()) { if (await _securitySettingsViewModel.biometricAuthenticated()) {
_securitySettingsViewModel _securitySettingsViewModel
@ -74,8 +65,6 @@ class SecurityBackupPage extends BasePage {
_securitySettingsViewModel _securitySettingsViewModel
.setAllowBiometricalAuthentication(isAuthenticatedSuccessfully); .setAllowBiometricalAuthentication(isAuthenticatedSuccessfully);
} }
auth.close();
}); });
} else { } else {
_securitySettingsViewModel.setAllowBiometricalAuthentication(value); _securitySettingsViewModel.setAllowBiometricalAuthentication(value);

View file

@ -1,4 +1,4 @@
import 'package:cake_wallet/src/screens/auth/auth_page.dart'; import 'package:cake_wallet/core/auth_service.dart';
import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart'; import 'package:cake_wallet/src/widgets/alert_with_two_actions.dart';
import 'package:cake_wallet/utils/device_info.dart'; import 'package:cake_wallet/utils/device_info.dart';
import 'package:cake_wallet/utils/show_bar.dart'; import 'package:cake_wallet/utils/show_bar.dart';
@ -19,18 +19,21 @@ import 'package:flutter_slidable/flutter_slidable.dart';
import 'package:cake_wallet/wallet_type_utils.dart'; import 'package:cake_wallet/wallet_type_utils.dart';
class WalletListPage extends BasePage { class WalletListPage extends BasePage {
WalletListPage({required this.walletListViewModel}); WalletListPage({required this.walletListViewModel, required this.authService});
final WalletListViewModel walletListViewModel; final WalletListViewModel walletListViewModel;
final AuthService authService;
@override @override
Widget body(BuildContext context) => WalletListBody(walletListViewModel: walletListViewModel); Widget body(BuildContext context) =>
WalletListBody(walletListViewModel: walletListViewModel, authService: authService);
} }
class WalletListBody extends StatefulWidget { class WalletListBody extends StatefulWidget {
WalletListBody({required this.walletListViewModel}); WalletListBody({required this.walletListViewModel, required this.authService});
final WalletListViewModel walletListViewModel; final WalletListViewModel walletListViewModel;
final AuthService authService;
@override @override
WalletListBodyState createState() => WalletListBodyState(); WalletListBodyState createState() => WalletListBodyState();
@ -129,7 +132,8 @@ class WalletListBodyState extends State<WalletListBody> {
fontSize: 22, fontSize: 22,
fontWeight: FontWeight.w500, fontWeight: FontWeight.w500,
color: Theme.of(context) color: Theme.of(context)
.primaryTextTheme.headline6! .primaryTextTheme
.headline6!
.color!), .color!),
) )
], ],
@ -201,18 +205,16 @@ class WalletListBodyState extends State<WalletListBody> {
} }
Future<void> _loadWallet(WalletListItem wallet) async { Future<void> _loadWallet(WalletListItem wallet) async {
if (await widget.walletListViewModel.checkIfAuthRequired()) { await widget.authService.authenticateAction(context,
await Navigator.of(context).pushNamed(Routes.auth, onAuthSuccess: (isAuthenticatedSuccessfully) async {
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) async {
if (!isAuthenticatedSuccessfully) { if (!isAuthenticatedSuccessfully) {
return; return;
} }
try { try {
auth.changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name)); changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name));
await widget.walletListViewModel.loadWallet(wallet); await widget.walletListViewModel.loadWallet(wallet);
auth.hideProgressText(); await hideProgressText();
auth.close();
// only pop the wallets route in mobile as it will go back to dashboard page // only pop the wallets route in mobile as it will go back to dashboard page
// in desktop platforms the navigation tree is different // in desktop platforms the navigation tree is different
if (DeviceInfo.instance.isMobile) { if (DeviceInfo.instance.isMobile) {
@ -220,42 +222,23 @@ class WalletListBodyState extends State<WalletListBody> {
Navigator.of(context).pop(); Navigator.of(context).pop();
}); });
} }
} catch (e) {
auth.changeProcessText(
S.of(context).wallet_list_failed_to_load(wallet.name, e.toString()));
}
});
} else {
try {
changeProcessText(S.of(context).wallet_list_loading_wallet(wallet.name));
await widget.walletListViewModel.loadWallet(wallet);
hideProgressText();
// only pop the wallets route in mobile as it will go back to dashboard page
// in desktop platforms the navigation tree is different
if (DeviceInfo.instance.isMobile) {
Navigator.of(context).pop();
}
} catch (e) { } catch (e) {
changeProcessText(S.of(context).wallet_list_failed_to_load(wallet.name, e.toString())); changeProcessText(S.of(context).wallet_list_failed_to_load(wallet.name, e.toString()));
} }
} });
} }
Future<void> _removeWallet(WalletListItem wallet) async { Future<void> _removeWallet(WalletListItem wallet) async {
if (widget.walletListViewModel.checkIfAuthRequired()) { widget.authService.authenticateAction(context,
await Navigator.of(context).pushNamed(Routes.auth, onAuthSuccess: (isAuthenticatedSuccessfully) async {
arguments: (bool isAuthenticatedSuccessfully, AuthPageState auth) async {
if (!isAuthenticatedSuccessfully) { if (!isAuthenticatedSuccessfully) {
return; return;
} }
_onSuccessfulAuth(wallet, auth); _onSuccessfulAuth(wallet);
}); });
} else {
_onSuccessfulAuth(wallet, null);
}
} }
void _onSuccessfulAuth(WalletListItem wallet, AuthPageState? auth) async { void _onSuccessfulAuth(WalletListItem wallet) async {
bool confirmed = false; bool confirmed = false;
await showPopUp<void>( await showPopUp<void>(
context: context, context: context,
@ -275,31 +258,23 @@ class WalletListBodyState extends State<WalletListBody> {
if (confirmed) { if (confirmed) {
try { try {
auth != null changeProcessText(S.of(context).wallet_list_removing_wallet(wallet.name));
? auth.changeProcessText(S.of(context).wallet_list_removing_wallet(wallet.name))
: changeProcessText(S.of(context).wallet_list_removing_wallet(wallet.name));
await widget.walletListViewModel.remove(wallet); await widget.walletListViewModel.remove(wallet);
hideProgressText(); hideProgressText();
} catch (e) { } catch (e) {
auth != null changeProcessText(
? auth.changeProcessText(
S.of(context).wallet_list_failed_to_remove(wallet.name, e.toString()),
)
: changeProcessText(
S.of(context).wallet_list_failed_to_remove(wallet.name, e.toString()), S.of(context).wallet_list_failed_to_remove(wallet.name, e.toString()),
); );
} }
} }
auth?.close();
} }
void changeProcessText(String text) { void changeProcessText(String text) {
_progressBar = createBar<void>(text, duration: null)..show(context); _progressBar = createBar<void>(text, duration: null)..show(context);
} }
void hideProgressText() { Future<void> hideProgressText() async {
Future.delayed(Duration(milliseconds: 50), () { await Future.delayed(Duration(milliseconds: 50), () {
_progressBar?.dismiss(); _progressBar?.dismiss();
_progressBar = null; _progressBar = null;
}); });