invoice/lib/redux/auth/auth_middleware.dart

166 lines
5.8 KiB
Dart

import 'package:flutter/material.dart';
import 'package:flutter_keychain/flutter_keychain.dart';
import 'package:invoiceninja_flutter/.env.dart';
import 'package:invoiceninja_flutter/constants.dart';
import 'package:invoiceninja_flutter/redux/app/app_actions.dart';
import 'package:invoiceninja_flutter/ui/auth/login_vm.dart';
import 'package:invoiceninja_flutter/utils/formatting.dart';
import 'package:redux/redux.dart';
import 'package:invoiceninja_flutter/redux/auth/auth_actions.dart';
import 'package:invoiceninja_flutter/redux/app/app_state.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:invoiceninja_flutter/data/repositories/auth_repository.dart';
List<Middleware<AppState>> createStoreAuthMiddleware([
AuthRepository repository = const AuthRepository(),
]) {
final loginInit = _createLoginInit();
final loginRequest = _createLoginRequest(repository);
final oauthRequest = _createOAuthRequest(repository);
final refreshRequest = _createRefreshRequest(repository);
return [
TypedMiddleware<AppState, LoadUserLogin>(loginInit),
TypedMiddleware<AppState, UserLoginRequest>(loginRequest),
TypedMiddleware<AppState, OAuthLoginRequest>(oauthRequest),
TypedMiddleware<AppState, RefreshData>(refreshRequest),
];
}
void _saveAuthLocal(dynamic action) async {
await FlutterKeychain.put(key: kKeychainEmail, value: action.sendEmail);
if (formatApiUrlReadable(action.url) != kAppUrl) {
await FlutterKeychain.put(
key: kKeychainUrl, value: formatApiUrlMachine(action.url));
await FlutterKeychain.put(key: kKeychainSecret, value: action.secret);
}
}
void _loadAuthLocal(Store<AppState> store, dynamic action) async {
final SharedPreferences prefs = await SharedPreferences.getInstance();
final String email = await FlutterKeychain.get(key: kKeychainEmail) ??
prefs.getString(kSharedPrefEmail) ??
Config.LOGIN_EMAIL;
final String url = await FlutterKeychain.get(key: kKeychainUrl) ??
formatApiUrlMachine(prefs.getString(kSharedPrefUrl) ?? Config.LOGIN_URL);
final String secret = await FlutterKeychain.get(key: kKeychainSecret) ??
prefs.getString(kSharedPrefSecret) ??
Config.LOGIN_SECRET;
store.dispatch(UserLoginLoaded(email, url, secret));
final bool enableDarkMode = prefs.getBool(kSharedPrefEnableDarkMode) ?? false;
final bool emailPayment = prefs.getBool(kSharedPrefEmailPayment) ?? false;
store.dispatch(UserSettingsChanged(
enableDarkMode: enableDarkMode, emailPayment: emailPayment));
}
Middleware<AppState> _createLoginInit() {
return (Store<AppState> store, dynamic action, NextDispatcher next) {
_loadAuthLocal(store, action);
Navigator.of(action.context).pushReplacementNamed(LoginScreen.route);
next(action);
};
}
Middleware<AppState> _createLoginRequest(AuthRepository repository) {
return (Store<AppState> store, dynamic action, NextDispatcher next) {
repository
.login(
email: action.sendEmail,
password: action.password,
url: action.url,
secret: action.secret,
platform: action.platform,
oneTimePassword: action.oneTimePassword)
.then((data) {
_saveAuthLocal(action);
if (_isVersionSupported(data.version)) {
store.dispatch(
LoadDataSuccess(completer: action.completer, loginResponse: data));
} else {
store.dispatch(UserLoginFailure(
'The minimum version is v$kMinMajorAppVersion.$kMinMinorAppVersion.$kMinPatchAppVersion'));
}
}).catchError((Object error) {
print(error);
if (error.toString().contains('No host specified in URI')) {
store.dispatch(UserLoginFailure('Please check the URL is correct'));
} else {
store.dispatch(UserLoginFailure(error.toString()));
}
});
next(action);
};
}
Middleware<AppState> _createOAuthRequest(AuthRepository repository) {
return (Store<AppState> store, dynamic action, NextDispatcher next) {
repository
.oauthLogin(
token: action.token,
url: action.url,
secret: action.secret,
platform: action.platform)
.then((data) {
_saveAuthLocal(action);
if (_isVersionSupported(data.version)) {
store.dispatch(
LoadDataSuccess(completer: action.completer, loginResponse: data));
} else {
store.dispatch(UserLoginFailure(
'The minimum version is v$kMinMajorAppVersion.$kMinMinorAppVersion.$kMinPatchAppVersion'));
}
}).catchError((Object error) {
print(error);
store.dispatch(UserLoginFailure(error.toString()));
});
next(action);
};
}
Middleware<AppState> _createRefreshRequest(AuthRepository repository) {
return (Store<AppState> store, dynamic action, NextDispatcher next) async {
next(action);
_loadAuthLocal(store, action);
final SharedPreferences prefs = await SharedPreferences.getInstance();
final String url = await FlutterKeychain.get(key: kKeychainUrl) ??
formatApiUrlMachine(prefs.getString(kSharedPrefUrl) ?? Config.LOGIN_URL);
final String token = await FlutterKeychain.get(key: getKeychainTokenKey());
repository
.refresh(url: url, token: token, platform: action.platform)
.then((data) {
store.dispatch(
LoadDataSuccess(completer: action.completer, loginResponse: data));
}).catchError((Object error) {
print(error);
store.dispatch(UserLoginFailure(error.toString()));
if (action.completer != null) {
action.completer.completeError(error);
}
});
};
}
bool _isVersionSupported(String version) {
final parts = version.split('.');
final int major = int.parse(parts[0]);
final int minor = int.parse(parts[1]);
final int patch = int.parse(parts[2]);
return major >= kMinMajorAppVersion &&
minor >= kMinMinorAppVersion &&
patch >= kMinPatchAppVersion;
}