2023-03-15 22:29:07 +01:00
|
|
|
import 'dart:io';
|
|
|
|
|
2023-04-25 22:19:23 -04:00
|
|
|
import 'package:device_info_plus/device_info_plus.dart';
|
2022-02-03 10:06:44 -06:00
|
|
|
import 'package:flutter/material.dart';
|
2023-04-28 16:01:03 -04:00
|
|
|
import 'package:flutter_udid/flutter_udid.dart';
|
2022-02-03 10:06:44 -06:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2024-05-02 15:59:14 -05:00
|
|
|
import 'package:immich_mobile/providers/album/album.provider.dart';
|
|
|
|
import 'package:immich_mobile/providers/album/shared_album.provider.dart';
|
2024-04-30 21:36:40 -05:00
|
|
|
import 'package:immich_mobile/entities/store.entity.dart';
|
|
|
|
import 'package:immich_mobile/models/authentication/authentication_state.model.dart';
|
|
|
|
import 'package:immich_mobile/entities/user.entity.dart';
|
2024-05-02 15:59:14 -05:00
|
|
|
import 'package:immich_mobile/providers/api.provider.dart';
|
|
|
|
import 'package:immich_mobile/providers/db.provider.dart';
|
|
|
|
import 'package:immich_mobile/services/api.service.dart';
|
2023-05-17 19:36:02 +02:00
|
|
|
import 'package:immich_mobile/utils/db.dart';
|
2023-03-03 23:38:30 +01:00
|
|
|
import 'package:immich_mobile/utils/hash.dart';
|
2023-05-17 19:36:02 +02:00
|
|
|
import 'package:isar/isar.dart';
|
2023-07-15 21:52:41 -04:00
|
|
|
import 'package:logging/logging.dart';
|
2022-07-13 07:23:48 -05:00
|
|
|
import 'package:openapi/api.dart';
|
2022-02-03 10:06:44 -06:00
|
|
|
|
|
|
|
class AuthenticationNotifier extends StateNotifier<AuthenticationState> {
|
2022-06-26 03:46:51 +09:00
|
|
|
AuthenticationNotifier(
|
2022-07-13 07:23:48 -05:00
|
|
|
this._apiService,
|
2023-05-17 19:36:02 +02:00
|
|
|
this._db,
|
2023-12-04 17:21:05 +01:00
|
|
|
this._ref,
|
2022-07-13 07:23:48 -05:00
|
|
|
) : super(
|
2022-02-03 10:06:44 -06:00
|
|
|
AuthenticationState(
|
|
|
|
deviceId: "",
|
|
|
|
userId: "",
|
|
|
|
userEmail: "",
|
2023-11-11 20:03:32 -05:00
|
|
|
name: '',
|
2022-05-28 22:35:45 -05:00
|
|
|
profileImagePath: '',
|
|
|
|
isAdmin: false,
|
2022-06-27 15:13:07 -05:00
|
|
|
shouldChangePassword: false,
|
2022-05-28 22:35:45 -05:00
|
|
|
isAuthenticated: false,
|
2022-02-03 10:06:44 -06:00
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2022-07-13 07:23:48 -05:00
|
|
|
final ApiService _apiService;
|
2023-05-17 19:36:02 +02:00
|
|
|
final Isar _db;
|
2023-12-04 17:21:05 +01:00
|
|
|
final StateNotifierProviderRef<AuthenticationNotifier, AuthenticationState>
|
|
|
|
_ref;
|
2023-07-27 20:05:27 -07:00
|
|
|
final _log = Logger("AuthenticationNotifier");
|
2022-07-13 07:23:48 -05:00
|
|
|
|
|
|
|
Future<bool> login(
|
|
|
|
String email,
|
|
|
|
String password,
|
2023-01-19 07:45:37 -08:00
|
|
|
String serverUrl,
|
2022-07-13 07:23:48 -05:00
|
|
|
) async {
|
2022-02-04 17:20:23 -06:00
|
|
|
try {
|
2023-01-19 07:45:37 -08:00
|
|
|
// Resolve API server endpoint from user provided serverUrl
|
|
|
|
await _apiService.resolveAndSetEndpoint(serverUrl);
|
2022-07-13 07:23:48 -05:00
|
|
|
await _apiService.serverInfoApi.pingServer();
|
2022-02-04 17:20:23 -06:00
|
|
|
} catch (e) {
|
2022-07-13 07:23:48 -05:00
|
|
|
debugPrint('Invalid Server Endpoint Url $e');
|
2022-02-03 10:06:44 -06:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sign-in request
|
2023-04-25 22:19:23 -04:00
|
|
|
DeviceInfoPlugin deviceInfoPlugin = DeviceInfoPlugin();
|
|
|
|
|
|
|
|
if (Platform.isIOS) {
|
|
|
|
var iosInfo = await deviceInfoPlugin.iosInfo;
|
|
|
|
_apiService.authenticationApi.apiClient
|
2024-01-27 16:14:32 +00:00
|
|
|
.addDefaultHeader('deviceModel', iosInfo.utsname.machine);
|
2023-04-25 22:19:23 -04:00
|
|
|
_apiService.authenticationApi.apiClient
|
|
|
|
.addDefaultHeader('deviceType', 'iOS');
|
|
|
|
} else {
|
|
|
|
var androidInfo = await deviceInfoPlugin.androidInfo;
|
|
|
|
_apiService.authenticationApi.apiClient
|
|
|
|
.addDefaultHeader('deviceModel', androidInfo.model);
|
|
|
|
_apiService.authenticationApi.apiClient
|
|
|
|
.addDefaultHeader('deviceType', 'Android');
|
|
|
|
}
|
|
|
|
|
2022-02-03 10:06:44 -06:00
|
|
|
try {
|
2022-07-13 07:23:48 -05:00
|
|
|
var loginResponse = await _apiService.authenticationApi.login(
|
|
|
|
LoginCredentialDto(
|
|
|
|
email: email,
|
|
|
|
password: password,
|
|
|
|
),
|
|
|
|
);
|
2022-02-03 10:06:44 -06:00
|
|
|
|
2022-07-13 07:23:48 -05:00
|
|
|
if (loginResponse == null) {
|
|
|
|
debugPrint('Login Response is null');
|
|
|
|
return false;
|
|
|
|
}
|
2022-02-03 10:06:44 -06:00
|
|
|
|
2022-11-20 11:43:10 -06:00
|
|
|
return setSuccessLoginInfo(
|
|
|
|
accessToken: loginResponse.accessToken,
|
2023-01-19 07:45:37 -08:00
|
|
|
serverUrl: serverUrl,
|
2022-02-03 10:06:44 -06:00
|
|
|
);
|
|
|
|
} catch (e) {
|
2022-07-13 07:23:48 -05:00
|
|
|
debugPrint("Error logging in $e");
|
2022-02-03 10:06:44 -06:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-15 21:52:41 -04:00
|
|
|
Future<void> logout() async {
|
|
|
|
var log = Logger('AuthenticationNotifier');
|
2023-02-20 21:43:39 -06:00
|
|
|
try {
|
2023-07-15 21:52:41 -04:00
|
|
|
String? userEmail = Store.tryGet(StoreKey.currentUser)?.email;
|
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
await _apiService.authenticationApi
|
2023-07-15 21:52:41 -04:00
|
|
|
.logout()
|
2023-08-09 22:01:16 -04:00
|
|
|
.then((_) => log.info("Logout was successful for $userEmail"))
|
2023-07-15 21:52:41 -04:00
|
|
|
.onError(
|
|
|
|
(error, stackTrace) =>
|
2024-02-24 04:38:57 +01:00
|
|
|
log.severe("Logout failed for $userEmail", error, stackTrace),
|
2023-07-15 21:52:41 -04:00
|
|
|
);
|
|
|
|
|
2023-02-20 21:43:39 -06:00
|
|
|
await Future.wait([
|
2023-05-17 19:36:02 +02:00
|
|
|
clearAssetsAndAlbums(_db),
|
2023-03-03 23:38:30 +01:00
|
|
|
Store.delete(StoreKey.currentUser),
|
2023-03-23 02:36:44 +01:00
|
|
|
Store.delete(StoreKey.accessToken),
|
2023-02-20 21:43:39 -06:00
|
|
|
]);
|
2023-12-04 17:21:05 +01:00
|
|
|
_ref.invalidate(albumProvider);
|
|
|
|
_ref.invalidate(sharedAlbumProvider);
|
2022-10-24 14:45:58 -05:00
|
|
|
|
2023-09-04 19:36:16 -04:00
|
|
|
state = state.copyWith(
|
|
|
|
deviceId: "",
|
|
|
|
userId: "",
|
|
|
|
userEmail: "",
|
2023-11-11 20:03:32 -05:00
|
|
|
name: '',
|
2023-09-04 19:36:16 -04:00
|
|
|
profileImagePath: '',
|
|
|
|
isAdmin: false,
|
|
|
|
shouldChangePassword: false,
|
|
|
|
isAuthenticated: false,
|
|
|
|
);
|
2024-02-24 04:38:57 +01:00
|
|
|
} catch (e, stack) {
|
|
|
|
log.severe('Logout failed', e, stack);
|
2023-02-20 21:43:39 -06:00
|
|
|
}
|
2022-02-03 10:06:44 -06:00
|
|
|
}
|
|
|
|
|
2022-05-28 22:35:45 -05:00
|
|
|
updateUserProfileImagePath(String path) {
|
|
|
|
state = state.copyWith(profileImagePath: path);
|
|
|
|
}
|
2022-06-27 15:13:07 -05:00
|
|
|
|
|
|
|
Future<bool> changePassword(String newPassword) async {
|
2022-07-13 07:23:48 -05:00
|
|
|
try {
|
2024-05-30 00:26:57 +02:00
|
|
|
await _apiService.usersApi.updateMyUser(
|
2024-05-26 18:15:52 -04:00
|
|
|
UserUpdateMeDto(
|
2022-07-13 07:23:48 -05:00
|
|
|
password: newPassword,
|
|
|
|
),
|
|
|
|
);
|
2022-06-27 15:13:07 -05:00
|
|
|
|
|
|
|
state = state.copyWith(shouldChangePassword: false);
|
2022-07-13 07:23:48 -05:00
|
|
|
|
2022-06-27 15:13:07 -05:00
|
|
|
return true;
|
2022-07-13 07:23:48 -05:00
|
|
|
} catch (e) {
|
|
|
|
debugPrint("Error changing password $e");
|
2022-06-27 15:13:07 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2022-11-20 11:43:10 -06:00
|
|
|
|
|
|
|
Future<bool> setSuccessLoginInfo({
|
|
|
|
required String accessToken,
|
2022-11-21 05:29:43 -06:00
|
|
|
required String serverUrl,
|
2022-11-20 11:43:10 -06:00
|
|
|
}) async {
|
|
|
|
_apiService.setAccessToken(accessToken);
|
2023-07-27 20:05:27 -07:00
|
|
|
|
|
|
|
// Get the deviceid from the store if it exists, otherwise generate a new one
|
|
|
|
String deviceId =
|
|
|
|
Store.tryGet(StoreKey.deviceId) ?? await FlutterUdid.consistentUdid;
|
|
|
|
|
|
|
|
bool shouldChangePassword = false;
|
2024-07-30 13:15:48 -05:00
|
|
|
User? user = Store.tryGet(StoreKey.currentUser);
|
2023-07-27 20:05:27 -07:00
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
UserAdminResponseDto? userResponse;
|
|
|
|
UserPreferencesResponseDto? userPreferences;
|
|
|
|
try {
|
|
|
|
final responses = await Future.wait([
|
|
|
|
_apiService.usersApi.getMyUser(),
|
|
|
|
_apiService.usersApi.getMyPreferences(),
|
|
|
|
]);
|
|
|
|
userResponse = responses[0] as UserAdminResponseDto;
|
|
|
|
userPreferences = responses[1] as UserPreferencesResponseDto;
|
|
|
|
} on ApiException catch (error, stackTrace) {
|
|
|
|
if (error.code == 401) {
|
|
|
|
_log.severe("Unauthorized access, token likely expired. Logging out.");
|
|
|
|
return false;
|
2023-03-15 22:29:07 +01:00
|
|
|
}
|
2024-07-30 13:15:48 -05:00
|
|
|
_log.severe(
|
|
|
|
"Error getting user information from the server [API EXCEPTION]",
|
|
|
|
stackTrace,
|
|
|
|
);
|
|
|
|
} catch (error, stackTrace) {
|
|
|
|
_log.severe(
|
|
|
|
"Error getting user information from the server [CATCH ALL]",
|
|
|
|
error,
|
|
|
|
stackTrace,
|
|
|
|
);
|
|
|
|
}
|
2022-11-20 11:43:10 -06:00
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
// If the user information is successfully retrieved, update the store
|
|
|
|
// Due to the flow of the code, this will always happen on first login
|
|
|
|
if (userResponse != null) {
|
|
|
|
Store.put(StoreKey.deviceId, deviceId);
|
|
|
|
Store.put(StoreKey.deviceIdHash, fastHash(deviceId));
|
|
|
|
Store.put(
|
|
|
|
StoreKey.currentUser,
|
|
|
|
User.fromUserDto(userResponse, userPreferences),
|
|
|
|
);
|
|
|
|
Store.put(StoreKey.serverUrl, serverUrl);
|
|
|
|
Store.put(StoreKey.accessToken, accessToken);
|
2023-07-27 20:05:27 -07:00
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
shouldChangePassword = userResponse.shouldChangePassword;
|
|
|
|
user = User.fromUserDto(userResponse, userPreferences);
|
|
|
|
} else {
|
|
|
|
_log.severe("Unable to get user information from the server.");
|
|
|
|
}
|
2023-07-27 20:05:27 -07:00
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
// If the user is null, the login was not successful
|
|
|
|
// and we don't have a local copy of the user from a prior successful login
|
|
|
|
if (user == null) {
|
|
|
|
return false;
|
2022-11-20 11:43:10 -06:00
|
|
|
}
|
2023-07-27 20:05:27 -07:00
|
|
|
|
|
|
|
state = state.copyWith(
|
|
|
|
isAuthenticated: true,
|
|
|
|
userId: user.id,
|
|
|
|
userEmail: user.email,
|
2023-11-11 20:03:32 -05:00
|
|
|
name: user.name,
|
2023-07-27 20:05:27 -07:00
|
|
|
profileImagePath: user.profileImagePath,
|
|
|
|
isAdmin: user.isAdmin,
|
|
|
|
shouldChangePassword: shouldChangePassword,
|
|
|
|
deviceId: deviceId,
|
|
|
|
);
|
|
|
|
|
2024-07-30 13:15:48 -05:00
|
|
|
return true;
|
2022-11-20 11:43:10 -06:00
|
|
|
}
|
2022-02-03 10:06:44 -06:00
|
|
|
}
|
|
|
|
|
2022-06-26 03:46:51 +09:00
|
|
|
final authenticationProvider =
|
|
|
|
StateNotifierProvider<AuthenticationNotifier, AuthenticationState>((ref) {
|
|
|
|
return AuthenticationNotifier(
|
2022-07-13 07:23:48 -05:00
|
|
|
ref.watch(apiServiceProvider),
|
2023-05-17 19:36:02 +02:00
|
|
|
ref.watch(dbProvider),
|
2023-12-04 17:21:05 +01:00
|
|
|
ref,
|
2022-06-26 03:46:51 +09:00
|
|
|
);
|
2022-02-03 10:06:44 -06:00
|
|
|
});
|