mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-23 19:16:09 +00:00
98 lines
2.7 KiB
Dart
98 lines
2.7 KiB
Dart
|
import 'dart:async';
|
||
|
import 'package:flutter/foundation.dart';
|
||
|
import 'package:shared_preferences/shared_preferences.dart';
|
||
|
import 'package:mobx/mobx.dart';
|
||
|
import 'package:cake_wallet/view_model/auth_state.dart';
|
||
|
import 'package:cake_wallet/core/auth_service.dart';
|
||
|
import 'package:cake_wallet/generated/i18n.dart';
|
||
|
|
||
|
part 'auth_view_model.g.dart';
|
||
|
|
||
|
class AuthViewModel = AuthViewModelBase with _$AuthViewModel;
|
||
|
|
||
|
abstract class AuthViewModelBase with Store {
|
||
|
AuthViewModelBase(
|
||
|
{@required this.authService, @required this.sharedPreferences}) {
|
||
|
state = AuthenticationStateInitial();
|
||
|
_failureCounter = 0;
|
||
|
}
|
||
|
|
||
|
static const maxFailedLogins = 3;
|
||
|
static const banTimeout = 180; // 3 mins
|
||
|
final banTimeoutKey = S.current.auth_store_ban_timeout;
|
||
|
|
||
|
final AuthService authService;
|
||
|
final SharedPreferences sharedPreferences;
|
||
|
|
||
|
@observable
|
||
|
AuthState state;
|
||
|
|
||
|
@observable
|
||
|
int _failureCounter;
|
||
|
|
||
|
@action
|
||
|
Future<void> auth({String password}) async {
|
||
|
state = AuthenticationStateInitial();
|
||
|
final _banDuration = banDuration();
|
||
|
|
||
|
if (_banDuration != null) {
|
||
|
state = AuthenticationBanned(
|
||
|
error: S.current.auth_store_banned_for +
|
||
|
'${_banDuration.inMinutes}' +
|
||
|
S.current.auth_store_banned_minutes);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
state = AuthenticationInProgress();
|
||
|
final isAuth = await authService.authenticate(password);
|
||
|
|
||
|
if (isAuth) {
|
||
|
state = AuthenticatedSuccessfully();
|
||
|
_failureCounter = 0;
|
||
|
} else {
|
||
|
_failureCounter += 1;
|
||
|
|
||
|
if (_failureCounter >= maxFailedLogins) {
|
||
|
final banDuration = await ban();
|
||
|
state = AuthenticationBanned(
|
||
|
error: S.current.auth_store_banned_for +
|
||
|
'${banDuration.inMinutes}' +
|
||
|
S.current.auth_store_banned_minutes);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
state =
|
||
|
AuthenticationFailure(error: S.current.auth_store_incorrect_password);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Duration banDuration() {
|
||
|
final unbanTimestamp = sharedPreferences.getInt(banTimeoutKey);
|
||
|
|
||
|
if (unbanTimestamp == null) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
final unbanTime = DateTime.fromMillisecondsSinceEpoch(unbanTimestamp);
|
||
|
final now = DateTime.now();
|
||
|
|
||
|
if (now.isAfter(unbanTime)) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
return Duration(milliseconds: unbanTimestamp - now.millisecondsSinceEpoch);
|
||
|
}
|
||
|
|
||
|
Future<Duration> ban() async {
|
||
|
final multiplier = _failureCounter - maxFailedLogins + 1;
|
||
|
final timeout = (multiplier * banTimeout) * 1000;
|
||
|
final unbanTimestamp = DateTime.now().millisecondsSinceEpoch + timeout;
|
||
|
await sharedPreferences.setInt(banTimeoutKey, unbanTimestamp);
|
||
|
|
||
|
return Duration(milliseconds: timeout);
|
||
|
}
|
||
|
|
||
|
@action
|
||
|
void biometricAuth() => state = AuthenticatedSuccessfully();
|
||
|
}
|