mirror of
https://github.com/immich-app/immich.git
synced 2025-01-25 05:02:46 +01:00
43e9529ce4
* feat(.well-known): add .well-known/immich to reference API endpoint
* feat(.well-known): make schema optional (defaults to https)
* adjust method comment to be a little less confusing
* fix casting issue with resovled url
* include when checking Well-known, update server hint
* add validation for login form's server url
* consolidate common process into resolveAndSetEndpoint
* fix missed prettier formatting
* revert translation changes
* update environment variable description, hopefully a bit clearer
* rename environment variable to IMMICH_API_URL_EXTERNAL
* comment out optional env variables
* fix(web): browser-side api client to include authorization token
* Revert "fix(web): browser-side api client to include authorization token"
This reverts commit 60e338938f
.
* remove multi-domain related changes
236 lines
7.1 KiB
Dart
236 lines
7.1 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:hive/hive.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/constants/hive_box.dart';
|
|
import 'package:immich_mobile/modules/album/services/album_cache.service.dart';
|
|
import 'package:immich_mobile/modules/home/services/asset_cache.service.dart';
|
|
import 'package:immich_mobile/modules/login/models/authentication_state.model.dart';
|
|
import 'package:immich_mobile/modules/login/models/hive_saved_login_info.model.dart';
|
|
import 'package:immich_mobile/modules/backup/services/backup.service.dart';
|
|
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
|
import 'package:immich_mobile/shared/services/api.service.dart';
|
|
import 'package:immich_mobile/shared/services/device_info.service.dart';
|
|
import 'package:openapi/api.dart';
|
|
|
|
class AuthenticationNotifier extends StateNotifier<AuthenticationState> {
|
|
AuthenticationNotifier(
|
|
this._deviceInfoService,
|
|
this._backupService,
|
|
this._apiService,
|
|
this._assetCacheService,
|
|
this._albumCacheService,
|
|
this._sharedAlbumCacheService,
|
|
) : super(
|
|
AuthenticationState(
|
|
deviceId: "",
|
|
deviceType: DeviceTypeEnum.ANDROID,
|
|
userId: "",
|
|
userEmail: "",
|
|
firstName: '',
|
|
lastName: '',
|
|
profileImagePath: '',
|
|
isAdmin: false,
|
|
shouldChangePassword: false,
|
|
isAuthenticated: false,
|
|
deviceInfo: DeviceInfoResponseDto(
|
|
id: 0,
|
|
userId: "",
|
|
deviceId: "",
|
|
deviceType: DeviceTypeEnum.ANDROID,
|
|
createdAt: "",
|
|
isAutoBackup: false,
|
|
),
|
|
),
|
|
);
|
|
|
|
final DeviceInfoService _deviceInfoService;
|
|
final BackupService _backupService;
|
|
final ApiService _apiService;
|
|
final AssetCacheService _assetCacheService;
|
|
final AlbumCacheService _albumCacheService;
|
|
final SharedAlbumCacheService _sharedAlbumCacheService;
|
|
|
|
Future<bool> login(
|
|
String email,
|
|
String password,
|
|
String serverUrl,
|
|
bool isSavedLoginInfo,
|
|
) async {
|
|
try {
|
|
// Resolve API server endpoint from user provided serverUrl
|
|
await _apiService.resolveAndSetEndpoint(serverUrl);
|
|
await _apiService.serverInfoApi.pingServer();
|
|
} catch (e) {
|
|
debugPrint('Invalid Server Endpoint Url $e');
|
|
return false;
|
|
}
|
|
|
|
// Make sign-in request
|
|
try {
|
|
var loginResponse = await _apiService.authenticationApi.login(
|
|
LoginCredentialDto(
|
|
email: email,
|
|
password: password,
|
|
),
|
|
);
|
|
|
|
if (loginResponse == null) {
|
|
debugPrint('Login Response is null');
|
|
return false;
|
|
}
|
|
|
|
return setSuccessLoginInfo(
|
|
accessToken: loginResponse.accessToken,
|
|
serverUrl: serverUrl,
|
|
isSavedLoginInfo: isSavedLoginInfo,
|
|
);
|
|
} catch (e) {
|
|
HapticFeedback.vibrate();
|
|
debugPrint("Error logging in $e");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<bool> logout() async {
|
|
state = state.copyWith(isAuthenticated: false);
|
|
await Future.wait([
|
|
Hive.box(userInfoBox).delete(accessTokenKey),
|
|
Hive.box(userInfoBox).delete(assetEtagKey),
|
|
_assetCacheService.invalidate(),
|
|
_albumCacheService.invalidate(),
|
|
_sharedAlbumCacheService.invalidate(),
|
|
]);
|
|
|
|
// Remove login info from local storage
|
|
var loginInfo =
|
|
Hive.box<HiveSavedLoginInfo>(hiveLoginInfoBox).get(savedLoginInfoKey);
|
|
if (loginInfo != null) {
|
|
loginInfo.email = "";
|
|
loginInfo.password = "";
|
|
loginInfo.isSaveLogin = false;
|
|
|
|
await Hive.box<HiveSavedLoginInfo>(hiveLoginInfoBox).put(
|
|
savedLoginInfoKey,
|
|
loginInfo,
|
|
);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
setAutoBackup(bool backupState) async {
|
|
var deviceInfo = await _deviceInfoService.getDeviceInfo();
|
|
var deviceId = deviceInfo["deviceId"];
|
|
|
|
DeviceTypeEnum deviceType = deviceInfo["deviceType"];
|
|
|
|
DeviceInfoResponseDto updatedDeviceInfo =
|
|
await _backupService.setAutoBackup(backupState, deviceId, deviceType);
|
|
|
|
state = state.copyWith(deviceInfo: updatedDeviceInfo);
|
|
}
|
|
|
|
updateUserProfileImagePath(String path) {
|
|
state = state.copyWith(profileImagePath: path);
|
|
}
|
|
|
|
Future<bool> changePassword(String newPassword) async {
|
|
try {
|
|
await _apiService.userApi.updateUser(
|
|
UpdateUserDto(
|
|
id: state.userId,
|
|
password: newPassword,
|
|
shouldChangePassword: false,
|
|
),
|
|
);
|
|
|
|
state = state.copyWith(shouldChangePassword: false);
|
|
|
|
return true;
|
|
} catch (e) {
|
|
debugPrint("Error changing password $e");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<bool> setSuccessLoginInfo({
|
|
required String accessToken,
|
|
required String serverUrl,
|
|
required bool isSavedLoginInfo,
|
|
}) async {
|
|
_apiService.setAccessToken(accessToken);
|
|
var userResponseDto = await _apiService.userApi.getMyUserInfo();
|
|
|
|
if (userResponseDto != null) {
|
|
var userInfoHiveBox = await Hive.openBox(userInfoBox);
|
|
var deviceInfo = await _deviceInfoService.getDeviceInfo();
|
|
userInfoHiveBox.put(deviceIdKey, deviceInfo["deviceId"]);
|
|
userInfoHiveBox.put(accessTokenKey, accessToken);
|
|
|
|
state = state.copyWith(
|
|
isAuthenticated: true,
|
|
userId: userResponseDto.id,
|
|
userEmail: userResponseDto.email,
|
|
firstName: userResponseDto.firstName,
|
|
lastName: userResponseDto.lastName,
|
|
profileImagePath: userResponseDto.profileImagePath,
|
|
isAdmin: userResponseDto.isAdmin,
|
|
shouldChangePassword: userResponseDto.shouldChangePassword,
|
|
deviceId: deviceInfo["deviceId"],
|
|
deviceType: deviceInfo["deviceType"],
|
|
);
|
|
|
|
if (isSavedLoginInfo) {
|
|
// Save login info to local storage
|
|
Hive.box<HiveSavedLoginInfo>(hiveLoginInfoBox).put(
|
|
savedLoginInfoKey,
|
|
HiveSavedLoginInfo(
|
|
email: "",
|
|
password: "",
|
|
isSaveLogin: true,
|
|
serverUrl: serverUrl,
|
|
accessToken: accessToken,
|
|
),
|
|
);
|
|
} else {
|
|
Hive.box<HiveSavedLoginInfo>(hiveLoginInfoBox)
|
|
.delete(savedLoginInfoKey);
|
|
}
|
|
}
|
|
|
|
// Register device info
|
|
try {
|
|
DeviceInfoResponseDto? deviceInfo =
|
|
await _apiService.deviceInfoApi.upsertDeviceInfo(
|
|
UpsertDeviceInfoDto(
|
|
deviceId: state.deviceId,
|
|
deviceType: state.deviceType,
|
|
),
|
|
);
|
|
|
|
if (deviceInfo == null) {
|
|
debugPrint('Device Info Response is null');
|
|
return false;
|
|
}
|
|
|
|
state = state.copyWith(deviceInfo: deviceInfo);
|
|
} catch (e) {
|
|
debugPrint("ERROR Register Device Info: $e");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
final authenticationProvider =
|
|
StateNotifierProvider<AuthenticationNotifier, AuthenticationState>((ref) {
|
|
return AuthenticationNotifier(
|
|
ref.watch(deviceInfoServiceProvider),
|
|
ref.watch(backupServiceProvider),
|
|
ref.watch(apiServiceProvider),
|
|
ref.watch(assetCacheServiceProvider),
|
|
ref.watch(albumCacheServiceProvider),
|
|
ref.watch(sharedAlbumCacheServiceProvider),
|
|
);
|
|
});
|