2023-05-25 05:52:43 +02:00
|
|
|
import 'package:collection/collection.dart';
|
2022-06-25 20:46:51 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2022-07-13 14:23:48 +02:00
|
|
|
import 'package:http/http.dart';
|
2022-05-29 05:35:45 +02:00
|
|
|
import 'package:image_picker/image_picker.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/services/partner.service.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/entities/store.entity.dart';
|
|
|
|
import 'package:immich_mobile/entities/user.entity.dart';
|
2024-05-02 22:59:14 +02: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';
|
|
|
|
import 'package:immich_mobile/services/sync.service.dart';
|
2023-05-25 05:52:43 +02:00
|
|
|
import 'package:immich_mobile/utils/diff.dart';
|
2023-03-03 23:38:30 +01:00
|
|
|
import 'package:isar/isar.dart';
|
2023-05-25 05:52:43 +02:00
|
|
|
import 'package:logging/logging.dart';
|
2022-07-13 14:23:48 +02:00
|
|
|
import 'package:openapi/api.dart';
|
2022-04-24 04:08:45 +02:00
|
|
|
|
2022-07-13 14:23:48 +02:00
|
|
|
final userServiceProvider = Provider(
|
|
|
|
(ref) => UserService(
|
|
|
|
ref.watch(apiServiceProvider),
|
2023-03-03 23:38:30 +01:00
|
|
|
ref.watch(dbProvider),
|
|
|
|
ref.watch(syncServiceProvider),
|
2023-05-25 05:52:43 +02:00
|
|
|
ref.watch(partnerServiceProvider),
|
2022-07-13 14:23:48 +02:00
|
|
|
),
|
|
|
|
);
|
2022-06-25 20:46:51 +02:00
|
|
|
|
2022-04-24 04:08:45 +02:00
|
|
|
class UserService {
|
2022-07-13 14:23:48 +02:00
|
|
|
final ApiService _apiService;
|
2023-03-03 23:38:30 +01:00
|
|
|
final Isar _db;
|
|
|
|
final SyncService _syncService;
|
2023-05-25 05:52:43 +02:00
|
|
|
final PartnerService _partnerService;
|
|
|
|
final Logger _log = Logger("UserService");
|
2022-04-24 04:08:45 +02:00
|
|
|
|
2023-05-25 05:52:43 +02:00
|
|
|
UserService(
|
|
|
|
this._apiService,
|
|
|
|
this._db,
|
|
|
|
this._syncService,
|
|
|
|
this._partnerService,
|
|
|
|
);
|
2022-04-24 04:08:45 +02:00
|
|
|
|
2024-05-27 00:15:52 +02:00
|
|
|
Future<List<User>?> _getAllUsers() async {
|
2022-07-13 14:23:48 +02:00
|
|
|
try {
|
2024-05-30 00:26:57 +02:00
|
|
|
final dto = await _apiService.usersApi.searchUsers();
|
2024-05-27 00:15:52 +02:00
|
|
|
return dto?.map(User.fromSimpleUserDto).toList();
|
2022-04-24 04:08:45 +02:00
|
|
|
} catch (e) {
|
2024-02-24 04:38:57 +01:00
|
|
|
_log.warning("Failed get all users", e);
|
2022-07-13 14:23:48 +02:00
|
|
|
return null;
|
2022-04-24 04:08:45 +02:00
|
|
|
}
|
|
|
|
}
|
2022-05-29 05:35:45 +02:00
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
Future<List<User>> getUsersInDb({bool self = false}) async {
|
|
|
|
if (self) {
|
|
|
|
return _db.users.where().findAll();
|
|
|
|
}
|
2023-03-23 02:36:44 +01:00
|
|
|
final int userId = Store.get(StoreKey.currentUser).isarId;
|
2023-03-03 23:38:30 +01:00
|
|
|
return _db.users.where().isarIdNotEqualTo(userId).findAll();
|
|
|
|
}
|
|
|
|
|
2022-07-13 14:23:48 +02:00
|
|
|
Future<CreateProfileImageResponseDto?> uploadProfileImage(XFile image) async {
|
2022-05-29 05:35:45 +02:00
|
|
|
try {
|
2024-05-30 00:26:57 +02:00
|
|
|
return await _apiService.usersApi.createProfileImage(
|
2022-07-13 14:23:48 +02:00
|
|
|
MultipartFile.fromBytes(
|
|
|
|
'file',
|
|
|
|
await image.readAsBytes(),
|
|
|
|
filename: image.name,
|
|
|
|
),
|
2022-05-29 05:35:45 +02:00
|
|
|
);
|
|
|
|
} catch (e) {
|
2024-02-24 04:38:57 +01:00
|
|
|
_log.warning("Failed to upload profile image", e);
|
2022-05-29 05:35:45 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2023-03-03 23:38:30 +01:00
|
|
|
|
2024-05-14 17:35:37 +02:00
|
|
|
Future<List<User>?> getUsersFromServer() async {
|
2024-05-27 00:15:52 +02:00
|
|
|
final List<User>? users = await _getAllUsers();
|
2023-05-25 05:52:43 +02:00
|
|
|
final List<User>? sharedBy =
|
2024-07-08 22:41:39 +02:00
|
|
|
await _partnerService.getPartners(PartnerDirection.by);
|
2023-05-25 05:52:43 +02:00
|
|
|
final List<User>? sharedWith =
|
2024-07-08 22:41:39 +02:00
|
|
|
await _partnerService.getPartners(PartnerDirection.with_);
|
2023-05-25 05:52:43 +02:00
|
|
|
|
|
|
|
if (users == null || sharedBy == null || sharedWith == null) {
|
|
|
|
_log.warning("Failed to refresh users");
|
2024-05-14 17:35:37 +02:00
|
|
|
return null;
|
2023-03-03 23:38:30 +01:00
|
|
|
}
|
2023-05-25 05:52:43 +02:00
|
|
|
|
|
|
|
users.sortBy((u) => u.id);
|
|
|
|
sharedBy.sortBy((u) => u.id);
|
|
|
|
sharedWith.sortBy((u) => u.id);
|
|
|
|
|
|
|
|
diffSortedListsSync(
|
|
|
|
users,
|
|
|
|
sharedBy,
|
|
|
|
compare: (User a, User b) => a.id.compareTo(b.id),
|
|
|
|
both: (User a, User b) => a.isPartnerSharedBy = true,
|
|
|
|
onlyFirst: (_) {},
|
|
|
|
onlySecond: (_) {},
|
|
|
|
);
|
|
|
|
|
|
|
|
diffSortedListsSync(
|
|
|
|
users,
|
|
|
|
sharedWith,
|
|
|
|
compare: (User a, User b) => a.id.compareTo(b.id),
|
2023-11-13 16:54:41 +01:00
|
|
|
both: (User a, User b) {
|
|
|
|
a.isPartnerSharedWith = true;
|
|
|
|
a.inTimeline = b.inTimeline;
|
|
|
|
return true;
|
|
|
|
},
|
2023-05-25 05:52:43 +02:00
|
|
|
onlyFirst: (_) {},
|
|
|
|
onlySecond: (_) {},
|
|
|
|
);
|
|
|
|
|
2024-05-14 17:35:37 +02:00
|
|
|
return users;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<bool> refreshUsers() async {
|
|
|
|
final users = await getUsersFromServer();
|
|
|
|
if (users == null) return false;
|
2023-03-03 23:38:30 +01:00
|
|
|
return _syncService.syncUsersFromServer(users);
|
|
|
|
}
|
2022-04-24 04:08:45 +02:00
|
|
|
}
|