mirror of
https://github.com/immich-app/immich.git
synced 2026-05-24 16:42:30 -04:00
Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 2042133ba7 | |||
| db9093f073 | |||
| fd7ddfef54 | |||
| 0975b1599c |
Generated
+6
-1
@@ -92,10 +92,12 @@ Class | Method | HTTP request | Description
|
|||||||
*AlbumsApi* | [**getAlbumMapMarkers**](doc//AlbumsApi.md#getalbummapmarkers) | **GET** /albums/{id}/map-markers | Retrieve album map markers
|
*AlbumsApi* | [**getAlbumMapMarkers**](doc//AlbumsApi.md#getalbummapmarkers) | **GET** /albums/{id}/map-markers | Retrieve album map markers
|
||||||
*AlbumsApi* | [**getAlbumStatistics**](doc//AlbumsApi.md#getalbumstatistics) | **GET** /albums/statistics | Retrieve album statistics
|
*AlbumsApi* | [**getAlbumStatistics**](doc//AlbumsApi.md#getalbumstatistics) | **GET** /albums/statistics | Retrieve album statistics
|
||||||
*AlbumsApi* | [**getAllAlbums**](doc//AlbumsApi.md#getallalbums) | **GET** /albums | List all albums
|
*AlbumsApi* | [**getAllAlbums**](doc//AlbumsApi.md#getallalbums) | **GET** /albums | List all albums
|
||||||
|
*AlbumsApi* | [**getOwnAlbumUser**](doc//AlbumsApi.md#getownalbumuser) | **GET** /albums/{id}/user/self | Get own sharing permissions
|
||||||
*AlbumsApi* | [**removeAssetFromAlbum**](doc//AlbumsApi.md#removeassetfromalbum) | **DELETE** /albums/{id}/assets | Remove assets from an album
|
*AlbumsApi* | [**removeAssetFromAlbum**](doc//AlbumsApi.md#removeassetfromalbum) | **DELETE** /albums/{id}/assets | Remove assets from an album
|
||||||
*AlbumsApi* | [**removeUserFromAlbum**](doc//AlbumsApi.md#removeuserfromalbum) | **DELETE** /albums/{id}/user/{userId} | Remove user from album
|
*AlbumsApi* | [**removeUserFromAlbum**](doc//AlbumsApi.md#removeuserfromalbum) | **DELETE** /albums/{id}/user/{userId} | Remove user from album
|
||||||
*AlbumsApi* | [**updateAlbumInfo**](doc//AlbumsApi.md#updatealbuminfo) | **PATCH** /albums/{id} | Update an album
|
*AlbumsApi* | [**updateAlbumInfo**](doc//AlbumsApi.md#updatealbuminfo) | **PATCH** /albums/{id} | Update an album
|
||||||
*AlbumsApi* | [**updateAlbumUser**](doc//AlbumsApi.md#updatealbumuser) | **PUT** /albums/{id}/user/{userId} | Update user role
|
*AlbumsApi* | [**updateAlbumUser**](doc//AlbumsApi.md#updatealbumuser) | **PUT** /albums/{id}/user/{userId} | Update user role
|
||||||
|
*AlbumsApi* | [**updateOwnAlbumUser**](doc//AlbumsApi.md#updateownalbumuser) | **PUT** /albums/{id}/user/self | Update own sharing permissions
|
||||||
*AssetsApi* | [**checkBulkUpload**](doc//AssetsApi.md#checkbulkupload) | **POST** /assets/bulk-upload-check | Check bulk upload
|
*AssetsApi* | [**checkBulkUpload**](doc//AssetsApi.md#checkbulkupload) | **POST** /assets/bulk-upload-check | Check bulk upload
|
||||||
*AssetsApi* | [**copyAsset**](doc//AssetsApi.md#copyasset) | **PUT** /assets/copy | Copy asset
|
*AssetsApi* | [**copyAsset**](doc//AssetsApi.md#copyasset) | **PUT** /assets/copy | Copy asset
|
||||||
*AssetsApi* | [**deleteAssetMetadata**](doc//AssetsApi.md#deleteassetmetadata) | **DELETE** /assets/{id}/metadata/{key} | Delete asset metadata by key
|
*AssetsApi* | [**deleteAssetMetadata**](doc//AssetsApi.md#deleteassetmetadata) | **DELETE** /assets/{id}/metadata/{key} | Delete asset metadata by key
|
||||||
@@ -452,7 +454,7 @@ Class | Method | HTTP request | Description
|
|||||||
- [MemoryStatisticsResponseDto](doc//MemoryStatisticsResponseDto.md)
|
- [MemoryStatisticsResponseDto](doc//MemoryStatisticsResponseDto.md)
|
||||||
- [MemoryType](doc//MemoryType.md)
|
- [MemoryType](doc//MemoryType.md)
|
||||||
- [MemoryUpdateDto](doc//MemoryUpdateDto.md)
|
- [MemoryUpdateDto](doc//MemoryUpdateDto.md)
|
||||||
- [MergePersonDto](doc//MergePersonDto.md)
|
- [MergeFaceClusterDto](doc//MergeFaceClusterDto.md)
|
||||||
- [MetadataSearchDto](doc//MetadataSearchDto.md)
|
- [MetadataSearchDto](doc//MetadataSearchDto.md)
|
||||||
- [MirrorAxis](doc//MirrorAxis.md)
|
- [MirrorAxis](doc//MirrorAxis.md)
|
||||||
- [MirrorParameters](doc//MirrorParameters.md)
|
- [MirrorParameters](doc//MirrorParameters.md)
|
||||||
@@ -544,6 +546,8 @@ Class | Method | HTTP request | Description
|
|||||||
- [SharedLinkType](doc//SharedLinkType.md)
|
- [SharedLinkType](doc//SharedLinkType.md)
|
||||||
- [SharedLinksResponse](doc//SharedLinksResponse.md)
|
- [SharedLinksResponse](doc//SharedLinksResponse.md)
|
||||||
- [SharedLinksUpdate](doc//SharedLinksUpdate.md)
|
- [SharedLinksUpdate](doc//SharedLinksUpdate.md)
|
||||||
|
- [SharingOptionsResponseDto](doc//SharingOptionsResponseDto.md)
|
||||||
|
- [SharingPermission](doc//SharingPermission.md)
|
||||||
- [SignUpDto](doc//SignUpDto.md)
|
- [SignUpDto](doc//SignUpDto.md)
|
||||||
- [SmartSearchDto](doc//SmartSearchDto.md)
|
- [SmartSearchDto](doc//SmartSearchDto.md)
|
||||||
- [SourceType](doc//SourceType.md)
|
- [SourceType](doc//SourceType.md)
|
||||||
@@ -643,6 +647,7 @@ Class | Method | HTTP request | Description
|
|||||||
- [UpdateAlbumUserDto](doc//UpdateAlbumUserDto.md)
|
- [UpdateAlbumUserDto](doc//UpdateAlbumUserDto.md)
|
||||||
- [UpdateAssetDto](doc//UpdateAssetDto.md)
|
- [UpdateAssetDto](doc//UpdateAssetDto.md)
|
||||||
- [UpdateLibraryDto](doc//UpdateLibraryDto.md)
|
- [UpdateLibraryDto](doc//UpdateLibraryDto.md)
|
||||||
|
- [UpdateSharingOptionsDto](doc//UpdateSharingOptionsDto.md)
|
||||||
- [UsageByUserDto](doc//UsageByUserDto.md)
|
- [UsageByUserDto](doc//UsageByUserDto.md)
|
||||||
- [UserAdminCreateDto](doc//UserAdminCreateDto.md)
|
- [UserAdminCreateDto](doc//UserAdminCreateDto.md)
|
||||||
- [UserAdminDeleteDto](doc//UserAdminDeleteDto.md)
|
- [UserAdminDeleteDto](doc//UserAdminDeleteDto.md)
|
||||||
|
|||||||
Generated
+4
-1
@@ -198,7 +198,7 @@ part 'model/memory_search_order.dart';
|
|||||||
part 'model/memory_statistics_response_dto.dart';
|
part 'model/memory_statistics_response_dto.dart';
|
||||||
part 'model/memory_type.dart';
|
part 'model/memory_type.dart';
|
||||||
part 'model/memory_update_dto.dart';
|
part 'model/memory_update_dto.dart';
|
||||||
part 'model/merge_person_dto.dart';
|
part 'model/merge_face_cluster_dto.dart';
|
||||||
part 'model/metadata_search_dto.dart';
|
part 'model/metadata_search_dto.dart';
|
||||||
part 'model/mirror_axis.dart';
|
part 'model/mirror_axis.dart';
|
||||||
part 'model/mirror_parameters.dart';
|
part 'model/mirror_parameters.dart';
|
||||||
@@ -290,6 +290,8 @@ part 'model/shared_link_response_dto.dart';
|
|||||||
part 'model/shared_link_type.dart';
|
part 'model/shared_link_type.dart';
|
||||||
part 'model/shared_links_response.dart';
|
part 'model/shared_links_response.dart';
|
||||||
part 'model/shared_links_update.dart';
|
part 'model/shared_links_update.dart';
|
||||||
|
part 'model/sharing_options_response_dto.dart';
|
||||||
|
part 'model/sharing_permission.dart';
|
||||||
part 'model/sign_up_dto.dart';
|
part 'model/sign_up_dto.dart';
|
||||||
part 'model/smart_search_dto.dart';
|
part 'model/smart_search_dto.dart';
|
||||||
part 'model/source_type.dart';
|
part 'model/source_type.dart';
|
||||||
@@ -389,6 +391,7 @@ part 'model/update_album_dto.dart';
|
|||||||
part 'model/update_album_user_dto.dart';
|
part 'model/update_album_user_dto.dart';
|
||||||
part 'model/update_asset_dto.dart';
|
part 'model/update_asset_dto.dart';
|
||||||
part 'model/update_library_dto.dart';
|
part 'model/update_library_dto.dart';
|
||||||
|
part 'model/update_sharing_options_dto.dart';
|
||||||
part 'model/usage_by_user_dto.dart';
|
part 'model/usage_by_user_dto.dart';
|
||||||
part 'model/user_admin_create_dto.dart';
|
part 'model/user_admin_create_dto.dart';
|
||||||
part 'model/user_admin_delete_dto.dart';
|
part 'model/user_admin_delete_dto.dart';
|
||||||
|
|||||||
Generated
+110
@@ -580,6 +580,63 @@ class AlbumsApi {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get own sharing permissions
|
||||||
|
///
|
||||||
|
/// Get the own sharing permissions in a specific album.
|
||||||
|
///
|
||||||
|
/// Note: This method returns the HTTP [Response].
|
||||||
|
///
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
Future<Response> getOwnAlbumUserWithHttpInfo(String id,) async {
|
||||||
|
// ignore: prefer_const_declarations
|
||||||
|
final apiPath = r'/albums/{id}/user/self'
|
||||||
|
.replaceAll('{id}', id);
|
||||||
|
|
||||||
|
// ignore: prefer_final_locals
|
||||||
|
Object? postBody;
|
||||||
|
|
||||||
|
final queryParams = <QueryParam>[];
|
||||||
|
final headerParams = <String, String>{};
|
||||||
|
final formParams = <String, String>{};
|
||||||
|
|
||||||
|
const contentTypes = <String>[];
|
||||||
|
|
||||||
|
|
||||||
|
return apiClient.invokeAPI(
|
||||||
|
apiPath,
|
||||||
|
'GET',
|
||||||
|
queryParams,
|
||||||
|
postBody,
|
||||||
|
headerParams,
|
||||||
|
formParams,
|
||||||
|
contentTypes.isEmpty ? null : contentTypes.first,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get own sharing permissions
|
||||||
|
///
|
||||||
|
/// Get the own sharing permissions in a specific album.
|
||||||
|
///
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
Future<SharingOptionsResponseDto?> getOwnAlbumUser(String id,) async {
|
||||||
|
final response = await getOwnAlbumUserWithHttpInfo(id,);
|
||||||
|
if (response.statusCode >= HttpStatus.badRequest) {
|
||||||
|
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
||||||
|
}
|
||||||
|
// When a remote server returns no body with a status of 204, we shall not decode it.
|
||||||
|
// At the time of writing this, `dart:convert` will throw an "Unexpected end of input"
|
||||||
|
// FormatException when trying to decode an empty string.
|
||||||
|
if (response.body.isNotEmpty && response.statusCode != HttpStatus.noContent) {
|
||||||
|
return await apiClient.deserializeAsync(await _decodeBodyBytes(response), 'SharingOptionsResponseDto',) as SharingOptionsResponseDto;
|
||||||
|
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
/// Remove assets from an album
|
/// Remove assets from an album
|
||||||
///
|
///
|
||||||
/// Remove multiple assets from a specific album by its ID.
|
/// Remove multiple assets from a specific album by its ID.
|
||||||
@@ -816,4 +873,57 @@ class AlbumsApi {
|
|||||||
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Update own sharing permissions
|
||||||
|
///
|
||||||
|
/// Change the own sharing permissions in a specific album.
|
||||||
|
///
|
||||||
|
/// Note: This method returns the HTTP [Response].
|
||||||
|
///
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
///
|
||||||
|
/// * [UpdateSharingOptionsDto] updateSharingOptionsDto (required):
|
||||||
|
Future<Response> updateOwnAlbumUserWithHttpInfo(String id, UpdateSharingOptionsDto updateSharingOptionsDto,) async {
|
||||||
|
// ignore: prefer_const_declarations
|
||||||
|
final apiPath = r'/albums/{id}/user/self'
|
||||||
|
.replaceAll('{id}', id);
|
||||||
|
|
||||||
|
// ignore: prefer_final_locals
|
||||||
|
Object? postBody = updateSharingOptionsDto;
|
||||||
|
|
||||||
|
final queryParams = <QueryParam>[];
|
||||||
|
final headerParams = <String, String>{};
|
||||||
|
final formParams = <String, String>{};
|
||||||
|
|
||||||
|
const contentTypes = <String>['application/json'];
|
||||||
|
|
||||||
|
|
||||||
|
return apiClient.invokeAPI(
|
||||||
|
apiPath,
|
||||||
|
'PUT',
|
||||||
|
queryParams,
|
||||||
|
postBody,
|
||||||
|
headerParams,
|
||||||
|
formParams,
|
||||||
|
contentTypes.isEmpty ? null : contentTypes.first,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Update own sharing permissions
|
||||||
|
///
|
||||||
|
/// Change the own sharing permissions in a specific album.
|
||||||
|
///
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
///
|
||||||
|
/// * [UpdateSharingOptionsDto] updateSharingOptionsDto (required):
|
||||||
|
Future<void> updateOwnAlbumUser(String id, UpdateSharingOptionsDto updateSharingOptionsDto,) async {
|
||||||
|
final response = await updateOwnAlbumUserWithHttpInfo(id, updateSharingOptionsDto,);
|
||||||
|
if (response.statusCode >= HttpStatus.badRequest) {
|
||||||
|
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Generated
+6
-6
@@ -448,14 +448,14 @@ class PeopleApi {
|
|||||||
///
|
///
|
||||||
/// * [String] id (required):
|
/// * [String] id (required):
|
||||||
///
|
///
|
||||||
/// * [MergePersonDto] mergePersonDto (required):
|
/// * [MergeFaceClusterDto] mergeFaceClusterDto (required):
|
||||||
Future<Response> mergePersonWithHttpInfo(String id, MergePersonDto mergePersonDto,) async {
|
Future<Response> mergePersonWithHttpInfo(String id, MergeFaceClusterDto mergeFaceClusterDto,) async {
|
||||||
// ignore: prefer_const_declarations
|
// ignore: prefer_const_declarations
|
||||||
final apiPath = r'/people/{id}/merge'
|
final apiPath = r'/people/{id}/merge'
|
||||||
.replaceAll('{id}', id);
|
.replaceAll('{id}', id);
|
||||||
|
|
||||||
// ignore: prefer_final_locals
|
// ignore: prefer_final_locals
|
||||||
Object? postBody = mergePersonDto;
|
Object? postBody = mergeFaceClusterDto;
|
||||||
|
|
||||||
final queryParams = <QueryParam>[];
|
final queryParams = <QueryParam>[];
|
||||||
final headerParams = <String, String>{};
|
final headerParams = <String, String>{};
|
||||||
@@ -483,9 +483,9 @@ class PeopleApi {
|
|||||||
///
|
///
|
||||||
/// * [String] id (required):
|
/// * [String] id (required):
|
||||||
///
|
///
|
||||||
/// * [MergePersonDto] mergePersonDto (required):
|
/// * [MergeFaceClusterDto] mergeFaceClusterDto (required):
|
||||||
Future<List<BulkIdResponseDto>?> mergePerson(String id, MergePersonDto mergePersonDto,) async {
|
Future<List<BulkIdResponseDto>?> mergePerson(String id, MergeFaceClusterDto mergeFaceClusterDto,) async {
|
||||||
final response = await mergePersonWithHttpInfo(id, mergePersonDto,);
|
final response = await mergePersonWithHttpInfo(id, mergeFaceClusterDto,);
|
||||||
if (response.statusCode >= HttpStatus.badRequest) {
|
if (response.statusCode >= HttpStatus.badRequest) {
|
||||||
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
throw ApiException(response.statusCode, await _decodeBodyBytes(response));
|
||||||
}
|
}
|
||||||
|
|||||||
Generated
+8
-2
@@ -442,8 +442,8 @@ class ApiClient {
|
|||||||
return MemoryTypeTypeTransformer().decode(value);
|
return MemoryTypeTypeTransformer().decode(value);
|
||||||
case 'MemoryUpdateDto':
|
case 'MemoryUpdateDto':
|
||||||
return MemoryUpdateDto.fromJson(value);
|
return MemoryUpdateDto.fromJson(value);
|
||||||
case 'MergePersonDto':
|
case 'MergeFaceClusterDto':
|
||||||
return MergePersonDto.fromJson(value);
|
return MergeFaceClusterDto.fromJson(value);
|
||||||
case 'MetadataSearchDto':
|
case 'MetadataSearchDto':
|
||||||
return MetadataSearchDto.fromJson(value);
|
return MetadataSearchDto.fromJson(value);
|
||||||
case 'MirrorAxis':
|
case 'MirrorAxis':
|
||||||
@@ -626,6 +626,10 @@ class ApiClient {
|
|||||||
return SharedLinksResponse.fromJson(value);
|
return SharedLinksResponse.fromJson(value);
|
||||||
case 'SharedLinksUpdate':
|
case 'SharedLinksUpdate':
|
||||||
return SharedLinksUpdate.fromJson(value);
|
return SharedLinksUpdate.fromJson(value);
|
||||||
|
case 'SharingOptionsResponseDto':
|
||||||
|
return SharingOptionsResponseDto.fromJson(value);
|
||||||
|
case 'SharingPermission':
|
||||||
|
return SharingPermissionTypeTransformer().decode(value);
|
||||||
case 'SignUpDto':
|
case 'SignUpDto':
|
||||||
return SignUpDto.fromJson(value);
|
return SignUpDto.fromJson(value);
|
||||||
case 'SmartSearchDto':
|
case 'SmartSearchDto':
|
||||||
@@ -824,6 +828,8 @@ class ApiClient {
|
|||||||
return UpdateAssetDto.fromJson(value);
|
return UpdateAssetDto.fromJson(value);
|
||||||
case 'UpdateLibraryDto':
|
case 'UpdateLibraryDto':
|
||||||
return UpdateLibraryDto.fromJson(value);
|
return UpdateLibraryDto.fromJson(value);
|
||||||
|
case 'UpdateSharingOptionsDto':
|
||||||
|
return UpdateSharingOptionsDto.fromJson(value);
|
||||||
case 'UsageByUserDto':
|
case 'UsageByUserDto':
|
||||||
return UsageByUserDto.fromJson(value);
|
return UsageByUserDto.fromJson(value);
|
||||||
case 'UserAdminCreateDto':
|
case 'UserAdminCreateDto':
|
||||||
|
|||||||
Generated
+3
@@ -163,6 +163,9 @@ String parameterToString(dynamic value) {
|
|||||||
if (value is SharedLinkType) {
|
if (value is SharedLinkType) {
|
||||||
return SharedLinkTypeTypeTransformer().encode(value).toString();
|
return SharedLinkTypeTypeTransformer().encode(value).toString();
|
||||||
}
|
}
|
||||||
|
if (value is SharingPermission) {
|
||||||
|
return SharingPermissionTypeTransformer().encode(value).toString();
|
||||||
|
}
|
||||||
if (value is SourceType) {
|
if (value is SourceType) {
|
||||||
return SourceTypeTypeTransformer().encode(value).toString();
|
return SourceTypeTypeTransformer().encode(value).toString();
|
||||||
}
|
}
|
||||||
|
|||||||
+9
-1
@@ -37,6 +37,7 @@ class AssetResponseDto {
|
|||||||
this.owner,
|
this.owner,
|
||||||
required this.ownerId,
|
required this.ownerId,
|
||||||
this.people = const [],
|
this.people = const [],
|
||||||
|
this.permissions = const [],
|
||||||
this.resized,
|
this.resized,
|
||||||
this.stack,
|
this.stack,
|
||||||
this.tags = const [],
|
this.tags = const [],
|
||||||
@@ -140,6 +141,8 @@ class AssetResponseDto {
|
|||||||
|
|
||||||
List<PersonResponseDto> people;
|
List<PersonResponseDto> people;
|
||||||
|
|
||||||
|
List<SharingPermission> permissions;
|
||||||
|
|
||||||
/// Is resized
|
/// Is resized
|
||||||
///
|
///
|
||||||
/// Please note: This property should have been non-nullable! Since the specification file
|
/// Please note: This property should have been non-nullable! Since the specification file
|
||||||
@@ -195,6 +198,7 @@ class AssetResponseDto {
|
|||||||
other.owner == owner &&
|
other.owner == owner &&
|
||||||
other.ownerId == ownerId &&
|
other.ownerId == ownerId &&
|
||||||
_deepEquality.equals(other.people, people) &&
|
_deepEquality.equals(other.people, people) &&
|
||||||
|
_deepEquality.equals(other.permissions, permissions) &&
|
||||||
other.resized == resized &&
|
other.resized == resized &&
|
||||||
other.stack == stack &&
|
other.stack == stack &&
|
||||||
_deepEquality.equals(other.tags, tags) &&
|
_deepEquality.equals(other.tags, tags) &&
|
||||||
@@ -231,6 +235,7 @@ class AssetResponseDto {
|
|||||||
(owner == null ? 0 : owner!.hashCode) +
|
(owner == null ? 0 : owner!.hashCode) +
|
||||||
(ownerId.hashCode) +
|
(ownerId.hashCode) +
|
||||||
(people.hashCode) +
|
(people.hashCode) +
|
||||||
|
(permissions.hashCode) +
|
||||||
(resized == null ? 0 : resized!.hashCode) +
|
(resized == null ? 0 : resized!.hashCode) +
|
||||||
(stack == null ? 0 : stack!.hashCode) +
|
(stack == null ? 0 : stack!.hashCode) +
|
||||||
(tags.hashCode) +
|
(tags.hashCode) +
|
||||||
@@ -241,7 +246,7 @@ class AssetResponseDto {
|
|||||||
(width == null ? 0 : width!.hashCode);
|
(width == null ? 0 : width!.hashCode);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() => 'AssetResponseDto[checksum=$checksum, createdAt=$createdAt, duplicateId=$duplicateId, duration=$duration, exifInfo=$exifInfo, fileCreatedAt=$fileCreatedAt, fileModifiedAt=$fileModifiedAt, hasMetadata=$hasMetadata, height=$height, id=$id, isArchived=$isArchived, isEdited=$isEdited, isFavorite=$isFavorite, isOffline=$isOffline, isTrashed=$isTrashed, libraryId=$libraryId, livePhotoVideoId=$livePhotoVideoId, localDateTime=$localDateTime, originalFileName=$originalFileName, originalMimeType=$originalMimeType, originalPath=$originalPath, owner=$owner, ownerId=$ownerId, people=$people, resized=$resized, stack=$stack, tags=$tags, thumbhash=$thumbhash, type=$type, updatedAt=$updatedAt, visibility=$visibility, width=$width]';
|
String toString() => 'AssetResponseDto[checksum=$checksum, createdAt=$createdAt, duplicateId=$duplicateId, duration=$duration, exifInfo=$exifInfo, fileCreatedAt=$fileCreatedAt, fileModifiedAt=$fileModifiedAt, hasMetadata=$hasMetadata, height=$height, id=$id, isArchived=$isArchived, isEdited=$isEdited, isFavorite=$isFavorite, isOffline=$isOffline, isTrashed=$isTrashed, libraryId=$libraryId, livePhotoVideoId=$livePhotoVideoId, localDateTime=$localDateTime, originalFileName=$originalFileName, originalMimeType=$originalMimeType, originalPath=$originalPath, owner=$owner, ownerId=$ownerId, people=$people, permissions=$permissions, resized=$resized, stack=$stack, tags=$tags, thumbhash=$thumbhash, type=$type, updatedAt=$updatedAt, visibility=$visibility, width=$width]';
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
Map<String, dynamic> toJson() {
|
||||||
final json = <String, dynamic>{};
|
final json = <String, dynamic>{};
|
||||||
@@ -301,6 +306,7 @@ class AssetResponseDto {
|
|||||||
}
|
}
|
||||||
json[r'ownerId'] = this.ownerId;
|
json[r'ownerId'] = this.ownerId;
|
||||||
json[r'people'] = this.people;
|
json[r'people'] = this.people;
|
||||||
|
json[r'permissions'] = this.permissions;
|
||||||
if (this.resized != null) {
|
if (this.resized != null) {
|
||||||
json[r'resized'] = this.resized;
|
json[r'resized'] = this.resized;
|
||||||
} else {
|
} else {
|
||||||
@@ -361,6 +367,7 @@ class AssetResponseDto {
|
|||||||
owner: UserResponseDto.fromJson(json[r'owner']),
|
owner: UserResponseDto.fromJson(json[r'owner']),
|
||||||
ownerId: mapValueOfType<String>(json, r'ownerId')!,
|
ownerId: mapValueOfType<String>(json, r'ownerId')!,
|
||||||
people: PersonResponseDto.listFromJson(json[r'people']),
|
people: PersonResponseDto.listFromJson(json[r'people']),
|
||||||
|
permissions: SharingPermission.listFromJson(json[r'permissions']),
|
||||||
resized: mapValueOfType<bool>(json, r'resized'),
|
resized: mapValueOfType<bool>(json, r'resized'),
|
||||||
stack: AssetStackResponseDto.fromJson(json[r'stack']),
|
stack: AssetStackResponseDto.fromJson(json[r'stack']),
|
||||||
tags: TagResponseDto.listFromJson(json[r'tags']),
|
tags: TagResponseDto.listFromJson(json[r'tags']),
|
||||||
@@ -433,6 +440,7 @@ class AssetResponseDto {
|
|||||||
'originalFileName',
|
'originalFileName',
|
||||||
'originalPath',
|
'originalPath',
|
||||||
'ownerId',
|
'ownerId',
|
||||||
|
'permissions',
|
||||||
'thumbhash',
|
'thumbhash',
|
||||||
'type',
|
'type',
|
||||||
'updatedAt',
|
'updatedAt',
|
||||||
|
|||||||
Generated
+3
@@ -42,6 +42,7 @@ class JobName {
|
|||||||
static const databaseBackup = JobName._(r'DatabaseBackup');
|
static const databaseBackup = JobName._(r'DatabaseBackup');
|
||||||
static const facialRecognitionQueueAll = JobName._(r'FacialRecognitionQueueAll');
|
static const facialRecognitionQueueAll = JobName._(r'FacialRecognitionQueueAll');
|
||||||
static const facialRecognition = JobName._(r'FacialRecognition');
|
static const facialRecognition = JobName._(r'FacialRecognition');
|
||||||
|
static const facialRecognitionMerge = JobName._(r'FacialRecognitionMerge');
|
||||||
static const fileDelete = JobName._(r'FileDelete');
|
static const fileDelete = JobName._(r'FileDelete');
|
||||||
static const fileMigrationQueueAll = JobName._(r'FileMigrationQueueAll');
|
static const fileMigrationQueueAll = JobName._(r'FileMigrationQueueAll');
|
||||||
static const libraryDeleteCheck = JobName._(r'LibraryDeleteCheck');
|
static const libraryDeleteCheck = JobName._(r'LibraryDeleteCheck');
|
||||||
@@ -100,6 +101,7 @@ class JobName {
|
|||||||
databaseBackup,
|
databaseBackup,
|
||||||
facialRecognitionQueueAll,
|
facialRecognitionQueueAll,
|
||||||
facialRecognition,
|
facialRecognition,
|
||||||
|
facialRecognitionMerge,
|
||||||
fileDelete,
|
fileDelete,
|
||||||
fileMigrationQueueAll,
|
fileMigrationQueueAll,
|
||||||
libraryDeleteCheck,
|
libraryDeleteCheck,
|
||||||
@@ -193,6 +195,7 @@ class JobNameTypeTransformer {
|
|||||||
case r'DatabaseBackup': return JobName.databaseBackup;
|
case r'DatabaseBackup': return JobName.databaseBackup;
|
||||||
case r'FacialRecognitionQueueAll': return JobName.facialRecognitionQueueAll;
|
case r'FacialRecognitionQueueAll': return JobName.facialRecognitionQueueAll;
|
||||||
case r'FacialRecognition': return JobName.facialRecognition;
|
case r'FacialRecognition': return JobName.facialRecognition;
|
||||||
|
case r'FacialRecognitionMerge': return JobName.facialRecognitionMerge;
|
||||||
case r'FileDelete': return JobName.fileDelete;
|
case r'FileDelete': return JobName.fileDelete;
|
||||||
case r'FileMigrationQueueAll': return JobName.fileMigrationQueueAll;
|
case r'FileMigrationQueueAll': return JobName.fileMigrationQueueAll;
|
||||||
case r'LibraryDeleteCheck': return JobName.libraryDeleteCheck;
|
case r'LibraryDeleteCheck': return JobName.libraryDeleteCheck;
|
||||||
|
|||||||
+3
@@ -29,6 +29,7 @@ class ManualJobName {
|
|||||||
static const memoryCleanup = ManualJobName._(r'memory-cleanup');
|
static const memoryCleanup = ManualJobName._(r'memory-cleanup');
|
||||||
static const memoryCreate = ManualJobName._(r'memory-create');
|
static const memoryCreate = ManualJobName._(r'memory-create');
|
||||||
static const backupDatabase = ManualJobName._(r'backup-database');
|
static const backupDatabase = ManualJobName._(r'backup-database');
|
||||||
|
static const personGroupMerge = ManualJobName._(r'person-group-merge');
|
||||||
|
|
||||||
/// List of all possible values in this [enum][ManualJobName].
|
/// List of all possible values in this [enum][ManualJobName].
|
||||||
static const values = <ManualJobName>[
|
static const values = <ManualJobName>[
|
||||||
@@ -38,6 +39,7 @@ class ManualJobName {
|
|||||||
memoryCleanup,
|
memoryCleanup,
|
||||||
memoryCreate,
|
memoryCreate,
|
||||||
backupDatabase,
|
backupDatabase,
|
||||||
|
personGroupMerge,
|
||||||
];
|
];
|
||||||
|
|
||||||
static ManualJobName? fromJson(dynamic value) => ManualJobNameTypeTransformer().decode(value);
|
static ManualJobName? fromJson(dynamic value) => ManualJobNameTypeTransformer().decode(value);
|
||||||
@@ -82,6 +84,7 @@ class ManualJobNameTypeTransformer {
|
|||||||
case r'memory-cleanup': return ManualJobName.memoryCleanup;
|
case r'memory-cleanup': return ManualJobName.memoryCleanup;
|
||||||
case r'memory-create': return ManualJobName.memoryCreate;
|
case r'memory-create': return ManualJobName.memoryCreate;
|
||||||
case r'backup-database': return ManualJobName.backupDatabase;
|
case r'backup-database': return ManualJobName.backupDatabase;
|
||||||
|
case r'person-group-merge': return ManualJobName.personGroupMerge;
|
||||||
default:
|
default:
|
||||||
if (!allowNull) {
|
if (!allowNull) {
|
||||||
throw ArgumentError('Unknown enum value to decode: $data');
|
throw ArgumentError('Unknown enum value to decode: $data');
|
||||||
|
|||||||
+20
-20
@@ -10,17 +10,17 @@
|
|||||||
|
|
||||||
part of openapi.api;
|
part of openapi.api;
|
||||||
|
|
||||||
class MergePersonDto {
|
class MergeFaceClusterDto {
|
||||||
/// Returns a new [MergePersonDto] instance.
|
/// Returns a new [MergeFaceClusterDto] instance.
|
||||||
MergePersonDto({
|
MergeFaceClusterDto({
|
||||||
this.ids = const [],
|
this.ids = const [],
|
||||||
});
|
});
|
||||||
|
|
||||||
/// Person IDs to merge
|
/// Face cluster IDs to merge
|
||||||
List<String> ids;
|
List<String> ids;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
bool operator ==(Object other) => identical(this, other) || other is MergePersonDto &&
|
bool operator ==(Object other) => identical(this, other) || other is MergeFaceClusterDto &&
|
||||||
_deepEquality.equals(other.ids, ids);
|
_deepEquality.equals(other.ids, ids);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -29,7 +29,7 @@ class MergePersonDto {
|
|||||||
(ids.hashCode);
|
(ids.hashCode);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() => 'MergePersonDto[ids=$ids]';
|
String toString() => 'MergeFaceClusterDto[ids=$ids]';
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
Map<String, dynamic> toJson() {
|
||||||
final json = <String, dynamic>{};
|
final json = <String, dynamic>{};
|
||||||
@@ -37,15 +37,15 @@ class MergePersonDto {
|
|||||||
return json;
|
return json;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns a new [MergePersonDto] instance and imports its values from
|
/// Returns a new [MergeFaceClusterDto] instance and imports its values from
|
||||||
/// [value] if it's a [Map], null otherwise.
|
/// [value] if it's a [Map], null otherwise.
|
||||||
// ignore: prefer_constructors_over_static_methods
|
// ignore: prefer_constructors_over_static_methods
|
||||||
static MergePersonDto? fromJson(dynamic value) {
|
static MergeFaceClusterDto? fromJson(dynamic value) {
|
||||||
upgradeDto(value, "MergePersonDto");
|
upgradeDto(value, "MergeFaceClusterDto");
|
||||||
if (value is Map) {
|
if (value is Map) {
|
||||||
final json = value.cast<String, dynamic>();
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
return MergePersonDto(
|
return MergeFaceClusterDto(
|
||||||
ids: json[r'ids'] is Iterable
|
ids: json[r'ids'] is Iterable
|
||||||
? (json[r'ids'] as Iterable).cast<String>().toList(growable: false)
|
? (json[r'ids'] as Iterable).cast<String>().toList(growable: false)
|
||||||
: const [],
|
: const [],
|
||||||
@@ -54,11 +54,11 @@ class MergePersonDto {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
static List<MergePersonDto> listFromJson(dynamic json, {bool growable = false,}) {
|
static List<MergeFaceClusterDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
final result = <MergePersonDto>[];
|
final result = <MergeFaceClusterDto>[];
|
||||||
if (json is List && json.isNotEmpty) {
|
if (json is List && json.isNotEmpty) {
|
||||||
for (final row in json) {
|
for (final row in json) {
|
||||||
final value = MergePersonDto.fromJson(row);
|
final value = MergeFaceClusterDto.fromJson(row);
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
result.add(value);
|
result.add(value);
|
||||||
}
|
}
|
||||||
@@ -67,12 +67,12 @@ class MergePersonDto {
|
|||||||
return result.toList(growable: growable);
|
return result.toList(growable: growable);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Map<String, MergePersonDto> mapFromJson(dynamic json) {
|
static Map<String, MergeFaceClusterDto> mapFromJson(dynamic json) {
|
||||||
final map = <String, MergePersonDto>{};
|
final map = <String, MergeFaceClusterDto>{};
|
||||||
if (json is Map && json.isNotEmpty) {
|
if (json is Map && json.isNotEmpty) {
|
||||||
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
for (final entry in json.entries) {
|
for (final entry in json.entries) {
|
||||||
final value = MergePersonDto.fromJson(entry.value);
|
final value = MergeFaceClusterDto.fromJson(entry.value);
|
||||||
if (value != null) {
|
if (value != null) {
|
||||||
map[entry.key] = value;
|
map[entry.key] = value;
|
||||||
}
|
}
|
||||||
@@ -81,14 +81,14 @@ class MergePersonDto {
|
|||||||
return map;
|
return map;
|
||||||
}
|
}
|
||||||
|
|
||||||
// maps a json object with a list of MergePersonDto-objects as value to a dart map
|
// maps a json object with a list of MergeFaceClusterDto-objects as value to a dart map
|
||||||
static Map<String, List<MergePersonDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
static Map<String, List<MergeFaceClusterDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
final map = <String, List<MergePersonDto>>{};
|
final map = <String, List<MergeFaceClusterDto>>{};
|
||||||
if (json is Map && json.isNotEmpty) {
|
if (json is Map && json.isNotEmpty) {
|
||||||
// ignore: parameter_assignments
|
// ignore: parameter_assignments
|
||||||
json = json.cast<String, dynamic>();
|
json = json.cast<String, dynamic>();
|
||||||
for (final entry in json.entries) {
|
for (final entry in json.entries) {
|
||||||
map[entry.key] = MergePersonDto.listFromJson(entry.value, growable: growable,);
|
map[entry.key] = MergeFaceClusterDto.listFromJson(entry.value, growable: growable,);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return map;
|
return map;
|
||||||
+14
-1
@@ -15,6 +15,7 @@ class PersonResponseDto {
|
|||||||
PersonResponseDto({
|
PersonResponseDto({
|
||||||
required this.birthDate,
|
required this.birthDate,
|
||||||
this.color,
|
this.color,
|
||||||
|
required this.faceClusterId,
|
||||||
required this.id,
|
required this.id,
|
||||||
this.isFavorite,
|
this.isFavorite,
|
||||||
required this.isHidden,
|
required this.isHidden,
|
||||||
@@ -35,6 +36,9 @@ class PersonResponseDto {
|
|||||||
///
|
///
|
||||||
String? color;
|
String? color;
|
||||||
|
|
||||||
|
/// Face cluster ID
|
||||||
|
String? faceClusterId;
|
||||||
|
|
||||||
/// Person ID
|
/// Person ID
|
||||||
String id;
|
String id;
|
||||||
|
|
||||||
@@ -69,6 +73,7 @@ class PersonResponseDto {
|
|||||||
bool operator ==(Object other) => identical(this, other) || other is PersonResponseDto &&
|
bool operator ==(Object other) => identical(this, other) || other is PersonResponseDto &&
|
||||||
other.birthDate == birthDate &&
|
other.birthDate == birthDate &&
|
||||||
other.color == color &&
|
other.color == color &&
|
||||||
|
other.faceClusterId == faceClusterId &&
|
||||||
other.id == id &&
|
other.id == id &&
|
||||||
other.isFavorite == isFavorite &&
|
other.isFavorite == isFavorite &&
|
||||||
other.isHidden == isHidden &&
|
other.isHidden == isHidden &&
|
||||||
@@ -81,6 +86,7 @@ class PersonResponseDto {
|
|||||||
// ignore: unnecessary_parenthesis
|
// ignore: unnecessary_parenthesis
|
||||||
(birthDate == null ? 0 : birthDate!.hashCode) +
|
(birthDate == null ? 0 : birthDate!.hashCode) +
|
||||||
(color == null ? 0 : color!.hashCode) +
|
(color == null ? 0 : color!.hashCode) +
|
||||||
|
(faceClusterId == null ? 0 : faceClusterId!.hashCode) +
|
||||||
(id.hashCode) +
|
(id.hashCode) +
|
||||||
(isFavorite == null ? 0 : isFavorite!.hashCode) +
|
(isFavorite == null ? 0 : isFavorite!.hashCode) +
|
||||||
(isHidden.hashCode) +
|
(isHidden.hashCode) +
|
||||||
@@ -89,7 +95,7 @@ class PersonResponseDto {
|
|||||||
(updatedAt == null ? 0 : updatedAt!.hashCode);
|
(updatedAt == null ? 0 : updatedAt!.hashCode);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() => 'PersonResponseDto[birthDate=$birthDate, color=$color, id=$id, isFavorite=$isFavorite, isHidden=$isHidden, name=$name, thumbnailPath=$thumbnailPath, updatedAt=$updatedAt]';
|
String toString() => 'PersonResponseDto[birthDate=$birthDate, color=$color, faceClusterId=$faceClusterId, id=$id, isFavorite=$isFavorite, isHidden=$isHidden, name=$name, thumbnailPath=$thumbnailPath, updatedAt=$updatedAt]';
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
Map<String, dynamic> toJson() {
|
||||||
final json = <String, dynamic>{};
|
final json = <String, dynamic>{};
|
||||||
@@ -102,6 +108,11 @@ class PersonResponseDto {
|
|||||||
json[r'color'] = this.color;
|
json[r'color'] = this.color;
|
||||||
} else {
|
} else {
|
||||||
// json[r'color'] = null;
|
// json[r'color'] = null;
|
||||||
|
}
|
||||||
|
if (this.faceClusterId != null) {
|
||||||
|
json[r'faceClusterId'] = this.faceClusterId;
|
||||||
|
} else {
|
||||||
|
// json[r'faceClusterId'] = null;
|
||||||
}
|
}
|
||||||
json[r'id'] = this.id;
|
json[r'id'] = this.id;
|
||||||
if (this.isFavorite != null) {
|
if (this.isFavorite != null) {
|
||||||
@@ -131,6 +142,7 @@ class PersonResponseDto {
|
|||||||
return PersonResponseDto(
|
return PersonResponseDto(
|
||||||
birthDate: mapDateTime(json, r'birthDate', r''),
|
birthDate: mapDateTime(json, r'birthDate', r''),
|
||||||
color: mapValueOfType<String>(json, r'color'),
|
color: mapValueOfType<String>(json, r'color'),
|
||||||
|
faceClusterId: mapValueOfType<String>(json, r'faceClusterId'),
|
||||||
id: mapValueOfType<String>(json, r'id')!,
|
id: mapValueOfType<String>(json, r'id')!,
|
||||||
isFavorite: mapValueOfType<bool>(json, r'isFavorite'),
|
isFavorite: mapValueOfType<bool>(json, r'isFavorite'),
|
||||||
isHidden: mapValueOfType<bool>(json, r'isHidden')!,
|
isHidden: mapValueOfType<bool>(json, r'isHidden')!,
|
||||||
@@ -185,6 +197,7 @@ class PersonResponseDto {
|
|||||||
/// The list of required keys that must be present in a JSON.
|
/// The list of required keys that must be present in a JSON.
|
||||||
static const requiredKeys = <String>{
|
static const requiredKeys = <String>{
|
||||||
'birthDate',
|
'birthDate',
|
||||||
|
'faceClusterId',
|
||||||
'id',
|
'id',
|
||||||
'isHidden',
|
'isHidden',
|
||||||
'name',
|
'name',
|
||||||
|
|||||||
@@ -0,0 +1,107 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.18
|
||||||
|
|
||||||
|
// ignore_for_file: unused_element, unused_import
|
||||||
|
// ignore_for_file: always_put_required_named_parameters_first
|
||||||
|
// ignore_for_file: constant_identifier_names
|
||||||
|
// ignore_for_file: lines_longer_than_80_chars
|
||||||
|
|
||||||
|
part of openapi.api;
|
||||||
|
|
||||||
|
class SharingOptionsResponseDto {
|
||||||
|
/// Returns a new [SharingOptionsResponseDto] instance.
|
||||||
|
SharingOptionsResponseDto({
|
||||||
|
required this.inTimeline,
|
||||||
|
this.permissions = const [],
|
||||||
|
});
|
||||||
|
|
||||||
|
bool inTimeline;
|
||||||
|
|
||||||
|
List<SharingPermission> permissions;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) => identical(this, other) || other is SharingOptionsResponseDto &&
|
||||||
|
other.inTimeline == inTimeline &&
|
||||||
|
_deepEquality.equals(other.permissions, permissions);
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
// ignore: unnecessary_parenthesis
|
||||||
|
(inTimeline.hashCode) +
|
||||||
|
(permissions.hashCode);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'SharingOptionsResponseDto[inTimeline=$inTimeline, permissions=$permissions]';
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final json = <String, dynamic>{};
|
||||||
|
json[r'inTimeline'] = this.inTimeline;
|
||||||
|
json[r'permissions'] = this.permissions;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a new [SharingOptionsResponseDto] instance and imports its values from
|
||||||
|
/// [value] if it's a [Map], null otherwise.
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static SharingOptionsResponseDto? fromJson(dynamic value) {
|
||||||
|
upgradeDto(value, "SharingOptionsResponseDto");
|
||||||
|
if (value is Map) {
|
||||||
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
|
return SharingOptionsResponseDto(
|
||||||
|
inTimeline: mapValueOfType<bool>(json, r'inTimeline')!,
|
||||||
|
permissions: SharingPermission.listFromJson(json[r'permissions']),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<SharingOptionsResponseDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <SharingOptionsResponseDto>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = SharingOptionsResponseDto.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Map<String, SharingOptionsResponseDto> mapFromJson(dynamic json) {
|
||||||
|
final map = <String, SharingOptionsResponseDto>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
final value = SharingOptionsResponseDto.fromJson(entry.value);
|
||||||
|
if (value != null) {
|
||||||
|
map[entry.key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
// maps a json object with a list of SharingOptionsResponseDto-objects as value to a dart map
|
||||||
|
static Map<String, List<SharingOptionsResponseDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final map = <String, List<SharingOptionsResponseDto>>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
// ignore: parameter_assignments
|
||||||
|
json = json.cast<String, dynamic>();
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
map[entry.key] = SharingOptionsResponseDto.listFromJson(entry.value, growable: growable,);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The list of required keys that must be present in a JSON.
|
||||||
|
static const requiredKeys = <String>{
|
||||||
|
'inTimeline',
|
||||||
|
'permissions',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
+112
@@ -0,0 +1,112 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.18
|
||||||
|
|
||||||
|
// ignore_for_file: unused_element, unused_import
|
||||||
|
// ignore_for_file: always_put_required_named_parameters_first
|
||||||
|
// ignore_for_file: constant_identifier_names
|
||||||
|
// ignore_for_file: lines_longer_than_80_chars
|
||||||
|
|
||||||
|
part of openapi.api;
|
||||||
|
|
||||||
|
/// Sharing permission schema
|
||||||
|
class SharingPermission {
|
||||||
|
/// Instantiate a new enum with the provided [value].
|
||||||
|
const SharingPermission._(this.value);
|
||||||
|
|
||||||
|
/// The underlying value of this enum member.
|
||||||
|
final String value;
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => value;
|
||||||
|
|
||||||
|
String toJson() => value;
|
||||||
|
|
||||||
|
static const all = SharingPermission._(r'all');
|
||||||
|
static const assetPeriodRead = SharingPermission._(r'asset.read');
|
||||||
|
static const assetPeriodUpdate = SharingPermission._(r'asset.update');
|
||||||
|
static const assetPeriodEdit = SharingPermission._(r'asset.edit');
|
||||||
|
static const assetPeriodDelete = SharingPermission._(r'asset.delete');
|
||||||
|
static const assetPeriodShare = SharingPermission._(r'asset.share');
|
||||||
|
static const exifPeriodRead = SharingPermission._(r'exif.read');
|
||||||
|
static const personPeriodRead = SharingPermission._(r'person.read');
|
||||||
|
static const personPeriodUpdate = SharingPermission._(r'person.update');
|
||||||
|
static const personPeriodMerge = SharingPermission._(r'person.merge');
|
||||||
|
static const personPeriodDelete = SharingPermission._(r'person.delete');
|
||||||
|
|
||||||
|
/// List of all possible values in this [enum][SharingPermission].
|
||||||
|
static const values = <SharingPermission>[
|
||||||
|
all,
|
||||||
|
assetPeriodRead,
|
||||||
|
assetPeriodUpdate,
|
||||||
|
assetPeriodEdit,
|
||||||
|
assetPeriodDelete,
|
||||||
|
assetPeriodShare,
|
||||||
|
exifPeriodRead,
|
||||||
|
personPeriodRead,
|
||||||
|
personPeriodUpdate,
|
||||||
|
personPeriodMerge,
|
||||||
|
personPeriodDelete,
|
||||||
|
];
|
||||||
|
|
||||||
|
static SharingPermission? fromJson(dynamic value) => SharingPermissionTypeTransformer().decode(value);
|
||||||
|
|
||||||
|
static List<SharingPermission> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <SharingPermission>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = SharingPermission.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transformation class that can [encode] an instance of [SharingPermission] to String,
|
||||||
|
/// and [decode] dynamic data back to [SharingPermission].
|
||||||
|
class SharingPermissionTypeTransformer {
|
||||||
|
factory SharingPermissionTypeTransformer() => _instance ??= const SharingPermissionTypeTransformer._();
|
||||||
|
|
||||||
|
const SharingPermissionTypeTransformer._();
|
||||||
|
|
||||||
|
String encode(SharingPermission data) => data.value;
|
||||||
|
|
||||||
|
/// Decodes a [dynamic value][data] to a SharingPermission.
|
||||||
|
///
|
||||||
|
/// If [allowNull] is true and the [dynamic value][data] cannot be decoded successfully,
|
||||||
|
/// then null is returned. However, if [allowNull] is false and the [dynamic value][data]
|
||||||
|
/// cannot be decoded successfully, then an [UnimplementedError] is thrown.
|
||||||
|
///
|
||||||
|
/// The [allowNull] is very handy when an API changes and a new enum value is added or removed,
|
||||||
|
/// and users are still using an old app with the old code.
|
||||||
|
SharingPermission? decode(dynamic data, {bool allowNull = true}) {
|
||||||
|
if (data != null) {
|
||||||
|
switch (data) {
|
||||||
|
case r'all': return SharingPermission.all;
|
||||||
|
case r'asset.read': return SharingPermission.assetPeriodRead;
|
||||||
|
case r'asset.update': return SharingPermission.assetPeriodUpdate;
|
||||||
|
case r'asset.edit': return SharingPermission.assetPeriodEdit;
|
||||||
|
case r'asset.delete': return SharingPermission.assetPeriodDelete;
|
||||||
|
case r'asset.share': return SharingPermission.assetPeriodShare;
|
||||||
|
case r'exif.read': return SharingPermission.exifPeriodRead;
|
||||||
|
case r'person.read': return SharingPermission.personPeriodRead;
|
||||||
|
case r'person.update': return SharingPermission.personPeriodUpdate;
|
||||||
|
case r'person.merge': return SharingPermission.personPeriodMerge;
|
||||||
|
case r'person.delete': return SharingPermission.personPeriodDelete;
|
||||||
|
default:
|
||||||
|
if (!allowNull) {
|
||||||
|
throw ArgumentError('Unknown enum value to decode: $data');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Singleton [SharingPermissionTypeTransformer] instance.
|
||||||
|
static SharingPermissionTypeTransformer? _instance;
|
||||||
|
}
|
||||||
|
|
||||||
+14
-14
@@ -19,11 +19,11 @@ class SyncAssetFaceV2 {
|
|||||||
required this.boundingBoxY1,
|
required this.boundingBoxY1,
|
||||||
required this.boundingBoxY2,
|
required this.boundingBoxY2,
|
||||||
required this.deletedAt,
|
required this.deletedAt,
|
||||||
|
required this.faceClusterId,
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.imageHeight,
|
required this.imageHeight,
|
||||||
required this.imageWidth,
|
required this.imageWidth,
|
||||||
required this.isVisible,
|
required this.isVisible,
|
||||||
required this.personId,
|
|
||||||
required this.sourceType,
|
required this.sourceType,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -57,6 +57,9 @@ class SyncAssetFaceV2 {
|
|||||||
/// Face deleted at
|
/// Face deleted at
|
||||||
DateTime? deletedAt;
|
DateTime? deletedAt;
|
||||||
|
|
||||||
|
/// Person ID
|
||||||
|
String? faceClusterId;
|
||||||
|
|
||||||
/// Asset face ID
|
/// Asset face ID
|
||||||
String id;
|
String id;
|
||||||
|
|
||||||
@@ -75,9 +78,6 @@ class SyncAssetFaceV2 {
|
|||||||
/// Is the face visible in the asset
|
/// Is the face visible in the asset
|
||||||
bool isVisible;
|
bool isVisible;
|
||||||
|
|
||||||
/// Person ID
|
|
||||||
String? personId;
|
|
||||||
|
|
||||||
/// Source type
|
/// Source type
|
||||||
String sourceType;
|
String sourceType;
|
||||||
|
|
||||||
@@ -89,11 +89,11 @@ class SyncAssetFaceV2 {
|
|||||||
other.boundingBoxY1 == boundingBoxY1 &&
|
other.boundingBoxY1 == boundingBoxY1 &&
|
||||||
other.boundingBoxY2 == boundingBoxY2 &&
|
other.boundingBoxY2 == boundingBoxY2 &&
|
||||||
other.deletedAt == deletedAt &&
|
other.deletedAt == deletedAt &&
|
||||||
|
other.faceClusterId == faceClusterId &&
|
||||||
other.id == id &&
|
other.id == id &&
|
||||||
other.imageHeight == imageHeight &&
|
other.imageHeight == imageHeight &&
|
||||||
other.imageWidth == imageWidth &&
|
other.imageWidth == imageWidth &&
|
||||||
other.isVisible == isVisible &&
|
other.isVisible == isVisible &&
|
||||||
other.personId == personId &&
|
|
||||||
other.sourceType == sourceType;
|
other.sourceType == sourceType;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -105,15 +105,15 @@ class SyncAssetFaceV2 {
|
|||||||
(boundingBoxY1.hashCode) +
|
(boundingBoxY1.hashCode) +
|
||||||
(boundingBoxY2.hashCode) +
|
(boundingBoxY2.hashCode) +
|
||||||
(deletedAt == null ? 0 : deletedAt!.hashCode) +
|
(deletedAt == null ? 0 : deletedAt!.hashCode) +
|
||||||
|
(faceClusterId == null ? 0 : faceClusterId!.hashCode) +
|
||||||
(id.hashCode) +
|
(id.hashCode) +
|
||||||
(imageHeight.hashCode) +
|
(imageHeight.hashCode) +
|
||||||
(imageWidth.hashCode) +
|
(imageWidth.hashCode) +
|
||||||
(isVisible.hashCode) +
|
(isVisible.hashCode) +
|
||||||
(personId == null ? 0 : personId!.hashCode) +
|
|
||||||
(sourceType.hashCode);
|
(sourceType.hashCode);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String toString() => 'SyncAssetFaceV2[assetId=$assetId, boundingBoxX1=$boundingBoxX1, boundingBoxX2=$boundingBoxX2, boundingBoxY1=$boundingBoxY1, boundingBoxY2=$boundingBoxY2, deletedAt=$deletedAt, id=$id, imageHeight=$imageHeight, imageWidth=$imageWidth, isVisible=$isVisible, personId=$personId, sourceType=$sourceType]';
|
String toString() => 'SyncAssetFaceV2[assetId=$assetId, boundingBoxX1=$boundingBoxX1, boundingBoxX2=$boundingBoxX2, boundingBoxY1=$boundingBoxY1, boundingBoxY2=$boundingBoxY2, deletedAt=$deletedAt, faceClusterId=$faceClusterId, id=$id, imageHeight=$imageHeight, imageWidth=$imageWidth, isVisible=$isVisible, sourceType=$sourceType]';
|
||||||
|
|
||||||
Map<String, dynamic> toJson() {
|
Map<String, dynamic> toJson() {
|
||||||
final json = <String, dynamic>{};
|
final json = <String, dynamic>{};
|
||||||
@@ -128,16 +128,16 @@ class SyncAssetFaceV2 {
|
|||||||
: this.deletedAt!.toUtc().toIso8601String();
|
: this.deletedAt!.toUtc().toIso8601String();
|
||||||
} else {
|
} else {
|
||||||
// json[r'deletedAt'] = null;
|
// json[r'deletedAt'] = null;
|
||||||
|
}
|
||||||
|
if (this.faceClusterId != null) {
|
||||||
|
json[r'faceClusterId'] = this.faceClusterId;
|
||||||
|
} else {
|
||||||
|
// json[r'faceClusterId'] = null;
|
||||||
}
|
}
|
||||||
json[r'id'] = this.id;
|
json[r'id'] = this.id;
|
||||||
json[r'imageHeight'] = this.imageHeight;
|
json[r'imageHeight'] = this.imageHeight;
|
||||||
json[r'imageWidth'] = this.imageWidth;
|
json[r'imageWidth'] = this.imageWidth;
|
||||||
json[r'isVisible'] = this.isVisible;
|
json[r'isVisible'] = this.isVisible;
|
||||||
if (this.personId != null) {
|
|
||||||
json[r'personId'] = this.personId;
|
|
||||||
} else {
|
|
||||||
// json[r'personId'] = null;
|
|
||||||
}
|
|
||||||
json[r'sourceType'] = this.sourceType;
|
json[r'sourceType'] = this.sourceType;
|
||||||
return json;
|
return json;
|
||||||
}
|
}
|
||||||
@@ -157,11 +157,11 @@ class SyncAssetFaceV2 {
|
|||||||
boundingBoxY1: mapValueOfType<int>(json, r'boundingBoxY1')!,
|
boundingBoxY1: mapValueOfType<int>(json, r'boundingBoxY1')!,
|
||||||
boundingBoxY2: mapValueOfType<int>(json, r'boundingBoxY2')!,
|
boundingBoxY2: mapValueOfType<int>(json, r'boundingBoxY2')!,
|
||||||
deletedAt: mapDateTime(json, r'deletedAt', r'/^(?:(?:\\d\\d[2468][048]|\\d\\d[13579][26]|\\d\\d0[48]|[02468][048]00|[13579][26]00)-02-29|\\d{4}-(?:(?:0[13578]|1[02])-(?:0[1-9]|[12]\\d|3[01])|(?:0[469]|11)-(?:0[1-9]|[12]\\d|30)|(?:02)-(?:0[1-9]|1\\d|2[0-8])))T(?:(?:[01]\\d|2[0-3]):[0-5]\\d(?::[0-5]\\d(?:\\.\\d+)?)?(?:Z))$/'),
|
deletedAt: mapDateTime(json, r'deletedAt', r'/^(?:(?:\\d\\d[2468][048]|\\d\\d[13579][26]|\\d\\d0[48]|[02468][048]00|[13579][26]00)-02-29|\\d{4}-(?:(?:0[13578]|1[02])-(?:0[1-9]|[12]\\d|3[01])|(?:0[469]|11)-(?:0[1-9]|[12]\\d|30)|(?:02)-(?:0[1-9]|1\\d|2[0-8])))T(?:(?:[01]\\d|2[0-3]):[0-5]\\d(?::[0-5]\\d(?:\\.\\d+)?)?(?:Z))$/'),
|
||||||
|
faceClusterId: mapValueOfType<String>(json, r'faceClusterId'),
|
||||||
id: mapValueOfType<String>(json, r'id')!,
|
id: mapValueOfType<String>(json, r'id')!,
|
||||||
imageHeight: mapValueOfType<int>(json, r'imageHeight')!,
|
imageHeight: mapValueOfType<int>(json, r'imageHeight')!,
|
||||||
imageWidth: mapValueOfType<int>(json, r'imageWidth')!,
|
imageWidth: mapValueOfType<int>(json, r'imageWidth')!,
|
||||||
isVisible: mapValueOfType<bool>(json, r'isVisible')!,
|
isVisible: mapValueOfType<bool>(json, r'isVisible')!,
|
||||||
personId: mapValueOfType<String>(json, r'personId'),
|
|
||||||
sourceType: mapValueOfType<String>(json, r'sourceType')!,
|
sourceType: mapValueOfType<String>(json, r'sourceType')!,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -216,11 +216,11 @@ class SyncAssetFaceV2 {
|
|||||||
'boundingBoxY1',
|
'boundingBoxY1',
|
||||||
'boundingBoxY2',
|
'boundingBoxY2',
|
||||||
'deletedAt',
|
'deletedAt',
|
||||||
|
'faceClusterId',
|
||||||
'id',
|
'id',
|
||||||
'imageHeight',
|
'imageHeight',
|
||||||
'imageWidth',
|
'imageWidth',
|
||||||
'isVisible',
|
'isVisible',
|
||||||
'personId',
|
|
||||||
'sourceType',
|
'sourceType',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,107 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.18
|
||||||
|
|
||||||
|
// ignore_for_file: unused_element, unused_import
|
||||||
|
// ignore_for_file: always_put_required_named_parameters_first
|
||||||
|
// ignore_for_file: constant_identifier_names
|
||||||
|
// ignore_for_file: lines_longer_than_80_chars
|
||||||
|
|
||||||
|
part of openapi.api;
|
||||||
|
|
||||||
|
class UpdateSharingOptionsDto {
|
||||||
|
/// Returns a new [UpdateSharingOptionsDto] instance.
|
||||||
|
UpdateSharingOptionsDto({
|
||||||
|
required this.inTimeline,
|
||||||
|
this.permissions = const [],
|
||||||
|
});
|
||||||
|
|
||||||
|
bool inTimeline;
|
||||||
|
|
||||||
|
List<SharingPermission> permissions;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) => identical(this, other) || other is UpdateSharingOptionsDto &&
|
||||||
|
other.inTimeline == inTimeline &&
|
||||||
|
_deepEquality.equals(other.permissions, permissions);
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
// ignore: unnecessary_parenthesis
|
||||||
|
(inTimeline.hashCode) +
|
||||||
|
(permissions.hashCode);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'UpdateSharingOptionsDto[inTimeline=$inTimeline, permissions=$permissions]';
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final json = <String, dynamic>{};
|
||||||
|
json[r'inTimeline'] = this.inTimeline;
|
||||||
|
json[r'permissions'] = this.permissions;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a new [UpdateSharingOptionsDto] instance and imports its values from
|
||||||
|
/// [value] if it's a [Map], null otherwise.
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static UpdateSharingOptionsDto? fromJson(dynamic value) {
|
||||||
|
upgradeDto(value, "UpdateSharingOptionsDto");
|
||||||
|
if (value is Map) {
|
||||||
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
|
return UpdateSharingOptionsDto(
|
||||||
|
inTimeline: mapValueOfType<bool>(json, r'inTimeline')!,
|
||||||
|
permissions: SharingPermission.listFromJson(json[r'permissions']),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<UpdateSharingOptionsDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <UpdateSharingOptionsDto>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = UpdateSharingOptionsDto.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Map<String, UpdateSharingOptionsDto> mapFromJson(dynamic json) {
|
||||||
|
final map = <String, UpdateSharingOptionsDto>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
final value = UpdateSharingOptionsDto.fromJson(entry.value);
|
||||||
|
if (value != null) {
|
||||||
|
map[entry.key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
// maps a json object with a list of UpdateSharingOptionsDto-objects as value to a dart map
|
||||||
|
static Map<String, List<UpdateSharingOptionsDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final map = <String, List<UpdateSharingOptionsDto>>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
// ignore: parameter_assignments
|
||||||
|
json = json.cast<String, dynamic>();
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
map[entry.key] = UpdateSharingOptionsDto.listFromJson(entry.value, growable: growable,);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The list of required keys that must be present in a JSON.
|
||||||
|
static const requiredKeys = <String>{
|
||||||
|
'inTimeline',
|
||||||
|
'permissions',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
@@ -2277,6 +2277,121 @@
|
|||||||
"x-immich-permission": "album.read"
|
"x-immich-permission": "album.read"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"/albums/{id}/user/self": {
|
||||||
|
"get": {
|
||||||
|
"description": "Get the own sharing permissions in a specific album.",
|
||||||
|
"operationId": "getOwnAlbumUser",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "id",
|
||||||
|
"required": true,
|
||||||
|
"in": "path",
|
||||||
|
"schema": {
|
||||||
|
"format": "uuid",
|
||||||
|
"pattern": "^([0-9a-fA-F]{8}-[0-9a-fA-F]{4}-4[0-9a-fA-F]{3}-[89abAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12})$",
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/SharingOptionsResponseDto"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"description": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearer": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"cookie": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"api_key": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"summary": "Get own sharing permissions",
|
||||||
|
"tags": [
|
||||||
|
"Albums"
|
||||||
|
],
|
||||||
|
"x-immich-history": [
|
||||||
|
{
|
||||||
|
"version": "v3",
|
||||||
|
"state": "Added"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "v3",
|
||||||
|
"state": "Stable"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"x-immich-permission": "albumAsset.create",
|
||||||
|
"x-immich-state": "Stable"
|
||||||
|
},
|
||||||
|
"put": {
|
||||||
|
"description": "Change the own sharing permissions in a specific album.",
|
||||||
|
"operationId": "updateOwnAlbumUser",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "id",
|
||||||
|
"required": true,
|
||||||
|
"in": "path",
|
||||||
|
"schema": {
|
||||||
|
"format": "uuid",
|
||||||
|
"pattern": "^([0-9a-fA-F]{8}-[0-9a-fA-F]{4}-4[0-9a-fA-F]{3}-[89abAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12})$",
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"requestBody": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/UpdateSharingOptionsDto"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": true
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"204": {
|
||||||
|
"description": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearer": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"cookie": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"api_key": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"summary": "Update own sharing permissions",
|
||||||
|
"tags": [
|
||||||
|
"Albums"
|
||||||
|
],
|
||||||
|
"x-immich-history": [
|
||||||
|
{
|
||||||
|
"version": "v3",
|
||||||
|
"state": "Added"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"version": "v3",
|
||||||
|
"state": "Stable"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"x-immich-permission": "albumAsset.create",
|
||||||
|
"x-immich-state": "Stable"
|
||||||
|
}
|
||||||
|
},
|
||||||
"/albums/{id}/user/{userId}": {
|
"/albums/{id}/user/{userId}": {
|
||||||
"delete": {
|
"delete": {
|
||||||
"description": "Remove a user from an album. Use an ID of \"me\" to leave a shared album.",
|
"description": "Remove a user from an album. Use an ID of \"me\" to leave a shared album.",
|
||||||
@@ -8345,7 +8460,7 @@
|
|||||||
"content": {
|
"content": {
|
||||||
"application/json": {
|
"application/json": {
|
||||||
"schema": {
|
"schema": {
|
||||||
"$ref": "#/components/schemas/MergePersonDto"
|
"$ref": "#/components/schemas/MergeFaceClusterDto"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -16942,6 +17057,12 @@
|
|||||||
},
|
},
|
||||||
"type": "array"
|
"type": "array"
|
||||||
},
|
},
|
||||||
|
"permissions": {
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/SharingPermission"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
},
|
||||||
"resized": {
|
"resized": {
|
||||||
"description": "Is resized",
|
"description": "Is resized",
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
@@ -17013,6 +17134,7 @@
|
|||||||
"originalFileName",
|
"originalFileName",
|
||||||
"originalPath",
|
"originalPath",
|
||||||
"ownerId",
|
"ownerId",
|
||||||
|
"permissions",
|
||||||
"thumbhash",
|
"thumbhash",
|
||||||
"type",
|
"type",
|
||||||
"updatedAt",
|
"updatedAt",
|
||||||
@@ -18072,6 +18194,7 @@
|
|||||||
"DatabaseBackup",
|
"DatabaseBackup",
|
||||||
"FacialRecognitionQueueAll",
|
"FacialRecognitionQueueAll",
|
||||||
"FacialRecognition",
|
"FacialRecognition",
|
||||||
|
"FacialRecognitionMerge",
|
||||||
"FileDelete",
|
"FileDelete",
|
||||||
"FileMigrationQueueAll",
|
"FileMigrationQueueAll",
|
||||||
"LibraryDeleteCheck",
|
"LibraryDeleteCheck",
|
||||||
@@ -18481,7 +18604,8 @@
|
|||||||
"user-cleanup",
|
"user-cleanup",
|
||||||
"memory-cleanup",
|
"memory-cleanup",
|
||||||
"memory-create",
|
"memory-create",
|
||||||
"backup-database"
|
"backup-database",
|
||||||
|
"person-group-merge"
|
||||||
],
|
],
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
@@ -18807,10 +18931,10 @@
|
|||||||
},
|
},
|
||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
"MergePersonDto": {
|
"MergeFaceClusterDto": {
|
||||||
"properties": {
|
"properties": {
|
||||||
"ids": {
|
"ids": {
|
||||||
"description": "Person IDs to merge",
|
"description": "Face cluster IDs to merge",
|
||||||
"items": {
|
"items": {
|
||||||
"format": "uuid",
|
"format": "uuid",
|
||||||
"pattern": "^([0-9a-fA-F]{8}-[0-9a-fA-F]{4}-4[0-9a-fA-F]{3}-[89abAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12})$",
|
"pattern": "^([0-9a-fA-F]{8}-[0-9a-fA-F]{4}-4[0-9a-fA-F]{3}-[89abAB][0-9a-fA-F]{3}-[0-9a-fA-F]{12})$",
|
||||||
@@ -19835,6 +19959,11 @@
|
|||||||
],
|
],
|
||||||
"x-immich-state": "Stable"
|
"x-immich-state": "Stable"
|
||||||
},
|
},
|
||||||
|
"faceClusterId": {
|
||||||
|
"description": "Face cluster ID",
|
||||||
|
"nullable": true,
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"id": {
|
"id": {
|
||||||
"description": "Person ID",
|
"description": "Person ID",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
@@ -19885,6 +20014,7 @@
|
|||||||
},
|
},
|
||||||
"required": [
|
"required": [
|
||||||
"birthDate",
|
"birthDate",
|
||||||
|
"faceClusterId",
|
||||||
"id",
|
"id",
|
||||||
"isHidden",
|
"isHidden",
|
||||||
"name",
|
"name",
|
||||||
@@ -21797,6 +21927,41 @@
|
|||||||
},
|
},
|
||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
|
"SharingOptionsResponseDto": {
|
||||||
|
"properties": {
|
||||||
|
"inTimeline": {
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"permissions": {
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/SharingPermission"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"inTimeline",
|
||||||
|
"permissions"
|
||||||
|
],
|
||||||
|
"type": "object"
|
||||||
|
},
|
||||||
|
"SharingPermission": {
|
||||||
|
"description": "Sharing permission schema",
|
||||||
|
"enum": [
|
||||||
|
"all",
|
||||||
|
"asset.read",
|
||||||
|
"asset.update",
|
||||||
|
"asset.edit",
|
||||||
|
"asset.delete",
|
||||||
|
"asset.share",
|
||||||
|
"exif.read",
|
||||||
|
"person.read",
|
||||||
|
"person.update",
|
||||||
|
"person.merge",
|
||||||
|
"person.delete"
|
||||||
|
],
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"SignUpDto": {
|
"SignUpDto": {
|
||||||
"properties": {
|
"properties": {
|
||||||
"email": {
|
"email": {
|
||||||
@@ -22893,6 +23058,11 @@
|
|||||||
"pattern": "^(?:(?:\\d\\d[2468][048]|\\d\\d[13579][26]|\\d\\d0[48]|[02468][048]00|[13579][26]00)-02-29|\\d{4}-(?:(?:0[13578]|1[02])-(?:0[1-9]|[12]\\d|3[01])|(?:0[469]|11)-(?:0[1-9]|[12]\\d|30)|(?:02)-(?:0[1-9]|1\\d|2[0-8])))T(?:(?:[01]\\d|2[0-3]):[0-5]\\d(?::[0-5]\\d(?:\\.\\d+)?)?(?:Z))$",
|
"pattern": "^(?:(?:\\d\\d[2468][048]|\\d\\d[13579][26]|\\d\\d0[48]|[02468][048]00|[13579][26]00)-02-29|\\d{4}-(?:(?:0[13578]|1[02])-(?:0[1-9]|[12]\\d|3[01])|(?:0[469]|11)-(?:0[1-9]|[12]\\d|30)|(?:02)-(?:0[1-9]|1\\d|2[0-8])))T(?:(?:[01]\\d|2[0-3]):[0-5]\\d(?::[0-5]\\d(?:\\.\\d+)?)?(?:Z))$",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
},
|
},
|
||||||
|
"faceClusterId": {
|
||||||
|
"description": "Person ID",
|
||||||
|
"nullable": true,
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"id": {
|
"id": {
|
||||||
"description": "Asset face ID",
|
"description": "Asset face ID",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
@@ -22913,11 +23083,6 @@
|
|||||||
"description": "Is the face visible in the asset",
|
"description": "Is the face visible in the asset",
|
||||||
"type": "boolean"
|
"type": "boolean"
|
||||||
},
|
},
|
||||||
"personId": {
|
|
||||||
"description": "Person ID",
|
|
||||||
"nullable": true,
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"sourceType": {
|
"sourceType": {
|
||||||
"description": "Source type",
|
"description": "Source type",
|
||||||
"type": "string"
|
"type": "string"
|
||||||
@@ -22930,11 +23095,11 @@
|
|||||||
"boundingBoxY1",
|
"boundingBoxY1",
|
||||||
"boundingBoxY2",
|
"boundingBoxY2",
|
||||||
"deletedAt",
|
"deletedAt",
|
||||||
|
"faceClusterId",
|
||||||
"id",
|
"id",
|
||||||
"imageHeight",
|
"imageHeight",
|
||||||
"imageWidth",
|
"imageWidth",
|
||||||
"isVisible",
|
"isVisible",
|
||||||
"personId",
|
|
||||||
"sourceType"
|
"sourceType"
|
||||||
],
|
],
|
||||||
"type": "object"
|
"type": "object"
|
||||||
@@ -25426,6 +25591,24 @@
|
|||||||
},
|
},
|
||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
|
"UpdateSharingOptionsDto": {
|
||||||
|
"properties": {
|
||||||
|
"inTimeline": {
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"permissions": {
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/SharingPermission"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"inTimeline",
|
||||||
|
"permissions"
|
||||||
|
],
|
||||||
|
"type": "object"
|
||||||
|
},
|
||||||
"UsageByUserDto": {
|
"UsageByUserDto": {
|
||||||
"properties": {
|
"properties": {
|
||||||
"photos": {
|
"photos": {
|
||||||
|
|||||||
@@ -555,6 +555,14 @@ export type MapMarkerResponseDto = {
|
|||||||
/** State/Province name */
|
/** State/Province name */
|
||||||
state: string | null;
|
state: string | null;
|
||||||
};
|
};
|
||||||
|
export type SharingOptionsResponseDto = {
|
||||||
|
inTimeline: boolean;
|
||||||
|
permissions: SharingPermission[];
|
||||||
|
};
|
||||||
|
export type UpdateSharingOptionsDto = {
|
||||||
|
inTimeline: boolean;
|
||||||
|
permissions: SharingPermission[];
|
||||||
|
};
|
||||||
export type UpdateAlbumUserDto = {
|
export type UpdateAlbumUserDto = {
|
||||||
role: AlbumUserRole;
|
role: AlbumUserRole;
|
||||||
};
|
};
|
||||||
@@ -792,6 +800,8 @@ export type PersonResponseDto = {
|
|||||||
birthDate: string | null;
|
birthDate: string | null;
|
||||||
/** Person color (hex) */
|
/** Person color (hex) */
|
||||||
color?: string;
|
color?: string;
|
||||||
|
/** Face cluster ID */
|
||||||
|
faceClusterId: string | null;
|
||||||
/** Person ID */
|
/** Person ID */
|
||||||
id: string;
|
id: string;
|
||||||
/** Is favorite */
|
/** Is favorite */
|
||||||
@@ -875,6 +885,7 @@ export type AssetResponseDto = {
|
|||||||
/** Owner user ID */
|
/** Owner user ID */
|
||||||
ownerId: string;
|
ownerId: string;
|
||||||
people?: PersonResponseDto[];
|
people?: PersonResponseDto[];
|
||||||
|
permissions: SharingPermission[];
|
||||||
/** Is resized */
|
/** Is resized */
|
||||||
resized?: boolean;
|
resized?: boolean;
|
||||||
stack?: (AssetStackResponseDto) | null;
|
stack?: (AssetStackResponseDto) | null;
|
||||||
@@ -1460,8 +1471,8 @@ export type PersonUpdateDto = {
|
|||||||
/** Person name */
|
/** Person name */
|
||||||
name?: string;
|
name?: string;
|
||||||
};
|
};
|
||||||
export type MergePersonDto = {
|
export type MergeFaceClusterDto = {
|
||||||
/** Person IDs to merge */
|
/** Face cluster IDs to merge */
|
||||||
ids: string[];
|
ids: string[];
|
||||||
};
|
};
|
||||||
export type AssetFaceUpdateItem = {
|
export type AssetFaceUpdateItem = {
|
||||||
@@ -2922,6 +2933,8 @@ export type SyncAssetFaceV2 = {
|
|||||||
boundingBoxY2: number;
|
boundingBoxY2: number;
|
||||||
/** Face deleted at */
|
/** Face deleted at */
|
||||||
deletedAt: string | null;
|
deletedAt: string | null;
|
||||||
|
/** Person ID */
|
||||||
|
faceClusterId: string | null;
|
||||||
/** Asset face ID */
|
/** Asset face ID */
|
||||||
id: string;
|
id: string;
|
||||||
/** Image height */
|
/** Image height */
|
||||||
@@ -2930,8 +2943,6 @@ export type SyncAssetFaceV2 = {
|
|||||||
imageWidth: number;
|
imageWidth: number;
|
||||||
/** Is the face visible in the asset */
|
/** Is the face visible in the asset */
|
||||||
isVisible: boolean;
|
isVisible: boolean;
|
||||||
/** Person ID */
|
|
||||||
personId: string | null;
|
|
||||||
/** Source type */
|
/** Source type */
|
||||||
sourceType: string;
|
sourceType: string;
|
||||||
};
|
};
|
||||||
@@ -3727,6 +3738,32 @@ export function getAlbumMapMarkers({ id, key, slug }: {
|
|||||||
...opts
|
...opts
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* Get own sharing permissions
|
||||||
|
*/
|
||||||
|
export function getOwnAlbumUser({ id }: {
|
||||||
|
id: string;
|
||||||
|
}, opts?: Oazapfts.RequestOpts) {
|
||||||
|
return oazapfts.ok(oazapfts.fetchJson<{
|
||||||
|
status: 200;
|
||||||
|
data: SharingOptionsResponseDto;
|
||||||
|
}>(`/albums/${encodeURIComponent(id)}/user/self`, {
|
||||||
|
...opts
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Update own sharing permissions
|
||||||
|
*/
|
||||||
|
export function updateOwnAlbumUser({ id, updateSharingOptionsDto }: {
|
||||||
|
id: string;
|
||||||
|
updateSharingOptionsDto: UpdateSharingOptionsDto;
|
||||||
|
}, opts?: Oazapfts.RequestOpts) {
|
||||||
|
return oazapfts.ok(oazapfts.fetchText(`/albums/${encodeURIComponent(id)}/user/self`, oazapfts.json({
|
||||||
|
...opts,
|
||||||
|
method: "PUT",
|
||||||
|
body: updateSharingOptionsDto
|
||||||
|
})));
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* Remove user from album
|
* Remove user from album
|
||||||
*/
|
*/
|
||||||
@@ -5131,9 +5168,9 @@ export function updatePerson({ id, personUpdateDto }: {
|
|||||||
/**
|
/**
|
||||||
* Merge people
|
* Merge people
|
||||||
*/
|
*/
|
||||||
export function mergePerson({ id, mergePersonDto }: {
|
export function mergePerson({ id, mergeFaceClusterDto }: {
|
||||||
id: string;
|
id: string;
|
||||||
mergePersonDto: MergePersonDto;
|
mergeFaceClusterDto: MergeFaceClusterDto;
|
||||||
}, opts?: Oazapfts.RequestOpts) {
|
}, opts?: Oazapfts.RequestOpts) {
|
||||||
return oazapfts.ok(oazapfts.fetchJson<{
|
return oazapfts.ok(oazapfts.fetchJson<{
|
||||||
status: 200;
|
status: 200;
|
||||||
@@ -5141,7 +5178,7 @@ export function mergePerson({ id, mergePersonDto }: {
|
|||||||
}>(`/people/${encodeURIComponent(id)}/merge`, oazapfts.json({
|
}>(`/people/${encodeURIComponent(id)}/merge`, oazapfts.json({
|
||||||
...opts,
|
...opts,
|
||||||
method: "POST",
|
method: "POST",
|
||||||
body: mergePersonDto
|
body: mergeFaceClusterDto
|
||||||
})));
|
})));
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
@@ -6788,6 +6825,19 @@ export enum BulkIdErrorReason {
|
|||||||
Unknown = "unknown",
|
Unknown = "unknown",
|
||||||
Validation = "validation"
|
Validation = "validation"
|
||||||
}
|
}
|
||||||
|
export enum SharingPermission {
|
||||||
|
All = "all",
|
||||||
|
AssetRead = "asset.read",
|
||||||
|
AssetUpdate = "asset.update",
|
||||||
|
AssetEdit = "asset.edit",
|
||||||
|
AssetDelete = "asset.delete",
|
||||||
|
AssetShare = "asset.share",
|
||||||
|
ExifRead = "exif.read",
|
||||||
|
PersonRead = "person.read",
|
||||||
|
PersonUpdate = "person.update",
|
||||||
|
PersonMerge = "person.merge",
|
||||||
|
PersonDelete = "person.delete"
|
||||||
|
}
|
||||||
export enum Permission {
|
export enum Permission {
|
||||||
All = "all",
|
All = "all",
|
||||||
ActivityCreate = "activity.create",
|
ActivityCreate = "activity.create",
|
||||||
@@ -6995,7 +7045,8 @@ export enum ManualJobName {
|
|||||||
UserCleanup = "user-cleanup",
|
UserCleanup = "user-cleanup",
|
||||||
MemoryCleanup = "memory-cleanup",
|
MemoryCleanup = "memory-cleanup",
|
||||||
MemoryCreate = "memory-create",
|
MemoryCreate = "memory-create",
|
||||||
BackupDatabase = "backup-database"
|
BackupDatabase = "backup-database",
|
||||||
|
PersonGroupMerge = "person-group-merge"
|
||||||
}
|
}
|
||||||
export enum QueueName {
|
export enum QueueName {
|
||||||
ThumbnailGeneration = "thumbnailGeneration",
|
ThumbnailGeneration = "thumbnailGeneration",
|
||||||
@@ -7072,6 +7123,7 @@ export enum JobName {
|
|||||||
DatabaseBackup = "DatabaseBackup",
|
DatabaseBackup = "DatabaseBackup",
|
||||||
FacialRecognitionQueueAll = "FacialRecognitionQueueAll",
|
FacialRecognitionQueueAll = "FacialRecognitionQueueAll",
|
||||||
FacialRecognition = "FacialRecognition",
|
FacialRecognition = "FacialRecognition",
|
||||||
|
FacialRecognitionMerge = "FacialRecognitionMerge",
|
||||||
FileDelete = "FileDelete",
|
FileDelete = "FileDelete",
|
||||||
FileMigrationQueueAll = "FileMigrationQueueAll",
|
FileMigrationQueueAll = "FileMigrationQueueAll",
|
||||||
LibraryDeleteCheck = "LibraryDeleteCheck",
|
LibraryDeleteCheck = "LibraryDeleteCheck",
|
||||||
|
|||||||
+2
-2
@@ -88,8 +88,8 @@ ENV NODE_ENV=production \
|
|||||||
COPY --from=server /output/server-pruned ./server
|
COPY --from=server /output/server-pruned ./server
|
||||||
COPY --from=web /usr/src/app/web/build /build/www
|
COPY --from=web /usr/src/app/web/build /build/www
|
||||||
COPY --from=cli /output/cli-pruned ./cli
|
COPY --from=cli /output/cli-pruned ./cli
|
||||||
COPY --from=plugins /app/packages/plugin-core/dist /build/plugins/immich-core-plugin/dist
|
COPY --from=plugins /app/packages/plugin-core/dist /build/plugins/immich-plugin-core/dist
|
||||||
COPY --from=plugins /app/packages/plugin-core/manifest.json /build/plugins/immich-core-plugin/manifest.json
|
COPY --from=plugins /app/packages/plugin-core/manifest.json /build/plugins/immich-plugin-core/manifest.json
|
||||||
RUN ln -s ../../cli/bin/immich server/bin/immich
|
RUN ln -s ../../cli/bin/immich server/bin/immich
|
||||||
COPY LICENSE /licenses/LICENSE.txt
|
COPY LICENSE /licenses/LICENSE.txt
|
||||||
COPY LICENSE /LICENSE
|
COPY LICENSE /LICENSE
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import {
|
|||||||
GetAlbumsDto,
|
GetAlbumsDto,
|
||||||
UpdateAlbumDto,
|
UpdateAlbumDto,
|
||||||
UpdateAlbumUserDto,
|
UpdateAlbumUserDto,
|
||||||
|
UpdateSharingPermissionsDto as UpdateSharingOptionsDto,
|
||||||
} from 'src/dtos/album.dto';
|
} from 'src/dtos/album.dto';
|
||||||
import { BulkIdResponseDto, BulkIdsDto } from 'src/dtos/asset-ids.response.dto';
|
import { BulkIdResponseDto, BulkIdsDto } from 'src/dtos/asset-ids.response.dto';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
@@ -165,6 +166,33 @@ export class AlbumController {
|
|||||||
return this.service.addUsers(auth, id, dto);
|
return this.service.addUsers(auth, id, dto);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Get(':id/user/self')
|
||||||
|
@Authenticated({ permission: Permission.AlbumAssetCreate })
|
||||||
|
@Endpoint({
|
||||||
|
summary: 'Get own sharing permissions',
|
||||||
|
description: 'Get the own sharing permissions in a specific album.',
|
||||||
|
history: new HistoryBuilder().added('v3').stable('v3'),
|
||||||
|
})
|
||||||
|
getOwnAlbumUser(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto) {
|
||||||
|
return this.service.getSelf(auth, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Put(':id/user/self')
|
||||||
|
@Authenticated({ permission: Permission.AlbumAssetCreate })
|
||||||
|
@HttpCode(HttpStatus.NO_CONTENT)
|
||||||
|
@Endpoint({
|
||||||
|
summary: 'Update own sharing permissions',
|
||||||
|
description: 'Change the own sharing permissions in a specific album.',
|
||||||
|
history: new HistoryBuilder().added('v3').stable('v3'),
|
||||||
|
})
|
||||||
|
updateOwnAlbumUser(
|
||||||
|
@Auth() auth: AuthDto,
|
||||||
|
@Param() { id }: UUIDParamDto,
|
||||||
|
@Body() dto: UpdateSharingOptionsDto,
|
||||||
|
): Promise<void> {
|
||||||
|
return this.service.updateSelf(auth, id, dto);
|
||||||
|
}
|
||||||
|
|
||||||
@Put(':id/user/:userId')
|
@Put(':id/user/:userId')
|
||||||
@Authenticated({ permission: Permission.AlbumUserUpdate })
|
@Authenticated({ permission: Permission.AlbumUserUpdate })
|
||||||
@HttpCode(HttpStatus.NO_CONTENT)
|
@HttpCode(HttpStatus.NO_CONTENT)
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ import { BulkIdResponseDto, BulkIdsDto } from 'src/dtos/asset-ids.response.dto';
|
|||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import {
|
import {
|
||||||
AssetFaceUpdateDto,
|
AssetFaceUpdateDto,
|
||||||
MergePersonDto,
|
MergeFaceClusterDto,
|
||||||
PeopleResponseDto,
|
PeopleResponseDto,
|
||||||
PeopleUpdateDto,
|
PeopleUpdateDto,
|
||||||
PersonCreateDto,
|
PersonCreateDto,
|
||||||
@@ -182,7 +182,7 @@ export class PersonController {
|
|||||||
mergePerson(
|
mergePerson(
|
||||||
@Auth() auth: AuthDto,
|
@Auth() auth: AuthDto,
|
||||||
@Param() { id }: UUIDParamDto,
|
@Param() { id }: UUIDParamDto,
|
||||||
@Body() dto: MergePersonDto,
|
@Body() dto: MergeFaceClusterDto,
|
||||||
): Promise<BulkIdResponseDto[]> {
|
): Promise<BulkIdResponseDto[]> {
|
||||||
return this.service.mergePerson(auth, id, dto);
|
return this.service.mergePerson(auth, id, dto);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
MemoryType,
|
MemoryType,
|
||||||
Permission,
|
Permission,
|
||||||
SharedLinkType,
|
SharedLinkType,
|
||||||
|
SharingPermission,
|
||||||
SourceType,
|
SourceType,
|
||||||
UserAvatarColor,
|
UserAvatarColor,
|
||||||
UserStatus,
|
UserStatus,
|
||||||
@@ -209,6 +210,7 @@ export type Partner = {
|
|||||||
updatedAt: Date;
|
updatedAt: Date;
|
||||||
updateId: string;
|
updateId: string;
|
||||||
inTimeline: boolean;
|
inTimeline: boolean;
|
||||||
|
permissions: SharingPermission[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Place = {
|
export type Place = {
|
||||||
@@ -252,6 +254,7 @@ export type Person = {
|
|||||||
faceAssetId: string | null;
|
faceAssetId: string | null;
|
||||||
isHidden: boolean;
|
isHidden: boolean;
|
||||||
thumbnailPath: string;
|
thumbnailPath: string;
|
||||||
|
faceClusterId: string | null;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type AssetFace = {
|
export type AssetFace = {
|
||||||
@@ -264,7 +267,7 @@ export type AssetFace = {
|
|||||||
boundingBoxY2: number;
|
boundingBoxY2: number;
|
||||||
imageHeight: number;
|
imageHeight: number;
|
||||||
imageWidth: number;
|
imageWidth: number;
|
||||||
personId: string | null;
|
faceClusterId: string | null;
|
||||||
sourceType: SourceType;
|
sourceType: SourceType;
|
||||||
person?: ShallowDehydrateObject<Person> | null;
|
person?: ShallowDehydrateObject<Person> | null;
|
||||||
updatedAt: Date;
|
updatedAt: Date;
|
||||||
|
|||||||
@@ -3,8 +3,8 @@ import { createZodDto } from 'nestjs-zod';
|
|||||||
import { AlbumUser, AuthSharedLink } from 'src/database';
|
import { AlbumUser, AuthSharedLink } from 'src/database';
|
||||||
import { BulkIdErrorReasonSchema } from 'src/dtos/asset-ids.response.dto';
|
import { BulkIdErrorReasonSchema } from 'src/dtos/asset-ids.response.dto';
|
||||||
import { MapAsset } from 'src/dtos/asset-response.dto';
|
import { MapAsset } from 'src/dtos/asset-response.dto';
|
||||||
import { UserResponseSchema, mapUser } from 'src/dtos/user.dto';
|
import { mapUser, UserResponseSchema } from 'src/dtos/user.dto';
|
||||||
import { AlbumUserRole, AlbumUserRoleSchema, AssetOrder, AssetOrderSchema } from 'src/enum';
|
import { AlbumUserRole, AlbumUserRoleSchema, AssetOrder, AssetOrderSchema, SharingPermissionSchema } from 'src/enum';
|
||||||
import { MaybeDehydrated } from 'src/types';
|
import { MaybeDehydrated } from 'src/types';
|
||||||
import { asDateString } from 'src/utils/date';
|
import { asDateString } from 'src/utils/date';
|
||||||
import { stringToBool } from 'src/validation';
|
import { stringToBool } from 'src/validation';
|
||||||
@@ -63,6 +63,14 @@ const UpdateAlbumSchema = z
|
|||||||
})
|
})
|
||||||
.meta({ id: 'UpdateAlbumDto' });
|
.meta({ id: 'UpdateAlbumDto' });
|
||||||
|
|
||||||
|
const UpdateSharingOptionsSchema = z
|
||||||
|
.object({ inTimeline: z.boolean(), permissions: z.array(SharingPermissionSchema) })
|
||||||
|
.meta({ id: 'UpdateSharingOptionsDto' });
|
||||||
|
|
||||||
|
const SharingOptionsResponseSchema = z
|
||||||
|
.object({ inTimeline: z.boolean(), permissions: z.array(SharingPermissionSchema) })
|
||||||
|
.meta({ id: 'SharingOptionsResponseDto' });
|
||||||
|
|
||||||
const GetAlbumsSchema = z
|
const GetAlbumsSchema = z
|
||||||
.object({
|
.object({
|
||||||
isOwned: stringToBool
|
isOwned: stringToBool
|
||||||
@@ -147,6 +155,8 @@ export class UpdateAlbumDto extends createZodDto(UpdateAlbumSchema) {}
|
|||||||
export class GetAlbumsDto extends createZodDto(GetAlbumsSchema) {}
|
export class GetAlbumsDto extends createZodDto(GetAlbumsSchema) {}
|
||||||
export class AlbumStatisticsResponseDto extends createZodDto(AlbumStatisticsResponseSchema) {}
|
export class AlbumStatisticsResponseDto extends createZodDto(AlbumStatisticsResponseSchema) {}
|
||||||
export class UpdateAlbumUserDto extends createZodDto(UpdateAlbumUserSchema) {}
|
export class UpdateAlbumUserDto extends createZodDto(UpdateAlbumUserSchema) {}
|
||||||
|
export class UpdateSharingPermissionsDto extends createZodDto(UpdateSharingOptionsSchema) {}
|
||||||
|
export class SharingPermissionsResponseDto extends createZodDto(SharingOptionsResponseSchema) {}
|
||||||
export class AlbumResponseDto extends createZodDto(AlbumResponseSchema) {}
|
export class AlbumResponseDto extends createZodDto(AlbumResponseSchema) {}
|
||||||
class AlbumUserResponseDto extends createZodDto(AlbumUserResponseSchema) {}
|
class AlbumUserResponseDto extends createZodDto(AlbumUserResponseSchema) {}
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,8 @@ import {
|
|||||||
AssetVisibility,
|
AssetVisibility,
|
||||||
AssetVisibilitySchema,
|
AssetVisibilitySchema,
|
||||||
ChecksumAlgorithm,
|
ChecksumAlgorithm,
|
||||||
|
SharingPermission,
|
||||||
|
SharingPermissionSchema,
|
||||||
} from 'src/enum';
|
} from 'src/enum';
|
||||||
import { MaybeDehydrated } from 'src/types';
|
import { MaybeDehydrated } from 'src/types';
|
||||||
import { hexOrBufferToBase64 } from 'src/utils/bytes';
|
import { hexOrBufferToBase64 } from 'src/utils/bytes';
|
||||||
@@ -45,6 +47,7 @@ const SanitizedAssetResponseSchema = z
|
|||||||
hasMetadata: z.boolean().describe('Whether asset has metadata'),
|
hasMetadata: z.boolean().describe('Whether asset has metadata'),
|
||||||
width: z.int().min(0).nullable().describe('Asset width'),
|
width: z.int().min(0).nullable().describe('Asset width'),
|
||||||
height: z.int().min(0).nullable().describe('Asset height'),
|
height: z.int().min(0).nullable().describe('Asset height'),
|
||||||
|
permissions: z.array(SharingPermissionSchema),
|
||||||
})
|
})
|
||||||
.meta({ id: 'SanitizedAssetResponseDto' });
|
.meta({ id: 'SanitizedAssetResponseDto' });
|
||||||
|
|
||||||
@@ -113,6 +116,7 @@ export const AssetResponseSchema = SanitizedAssetResponseSchema.extend(
|
|||||||
.boolean()
|
.boolean()
|
||||||
.describe('Is edited')
|
.describe('Is edited')
|
||||||
.meta(new HistoryBuilder().added('v2.5.0').beta('v2.5.0').getExtensions()),
|
.meta(new HistoryBuilder().added('v2.5.0').beta('v2.5.0').getExtensions()),
|
||||||
|
permissions: z.array(SharingPermissionSchema),
|
||||||
}).shape,
|
}).shape,
|
||||||
).meta({ id: 'AssetResponseDto' });
|
).meta({ id: 'AssetResponseDto' });
|
||||||
|
|
||||||
@@ -154,6 +158,7 @@ export type MapAsset = {
|
|||||||
width: number | null;
|
width: number | null;
|
||||||
height: number | null;
|
height: number | null;
|
||||||
isEdited: boolean;
|
isEdited: boolean;
|
||||||
|
permissions?: { permission: SharingPermission }[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type AssetMapOptions = {
|
export type AssetMapOptions = {
|
||||||
@@ -192,8 +197,16 @@ const mapStack = (entity: { stack?: Stack | null }) => {
|
|||||||
|
|
||||||
export function mapAsset(entity: MaybeDehydrated<MapAsset>, options: AssetMapOptions = {}): AssetResponseDto {
|
export function mapAsset(entity: MaybeDehydrated<MapAsset>, options: AssetMapOptions = {}): AssetResponseDto {
|
||||||
const { stripMetadata = false, withStack = false } = options;
|
const { stripMetadata = false, withStack = false } = options;
|
||||||
|
const permissions =
|
||||||
|
options.auth?.user.id === entity.ownerId
|
||||||
|
? [SharingPermission.All]
|
||||||
|
: (entity.permissions?.map(({ permission }) => permission) ?? []);
|
||||||
|
|
||||||
if (stripMetadata) {
|
if (
|
||||||
|
stripMetadata ||
|
||||||
|
(entity.permissions &&
|
||||||
|
!(permissions.includes(SharingPermission.All) || permissions.includes(SharingPermission.ExifRead)))
|
||||||
|
) {
|
||||||
const sanitizedAssetResponse: SanitizedAssetResponseDto = {
|
const sanitizedAssetResponse: SanitizedAssetResponseDto = {
|
||||||
id: entity.id,
|
id: entity.id,
|
||||||
type: entity.type,
|
type: entity.type,
|
||||||
@@ -205,6 +218,7 @@ export function mapAsset(entity: MaybeDehydrated<MapAsset>, options: AssetMapOpt
|
|||||||
hasMetadata: false,
|
hasMetadata: false,
|
||||||
width: entity.width,
|
width: entity.width,
|
||||||
height: entity.height,
|
height: entity.height,
|
||||||
|
permissions,
|
||||||
};
|
};
|
||||||
return sanitizedAssetResponse as AssetResponseDto;
|
return sanitizedAssetResponse as AssetResponseDto;
|
||||||
}
|
}
|
||||||
@@ -242,5 +256,6 @@ export function mapAsset(entity: MaybeDehydrated<MapAsset>, options: AssetMapOpt
|
|||||||
width: entity.width,
|
width: entity.width,
|
||||||
height: entity.height,
|
height: entity.height,
|
||||||
isEdited: entity.isEdited,
|
isEdited: entity.isEdited,
|
||||||
|
permissions,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import { Selectable } from 'kysely';
|
|||||||
import { createZodDto } from 'nestjs-zod';
|
import { createZodDto } from 'nestjs-zod';
|
||||||
import { AssetFace, Person } from 'src/database';
|
import { AssetFace, Person } from 'src/database';
|
||||||
import { HistoryBuilder } from 'src/decorators';
|
import { HistoryBuilder } from 'src/decorators';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
|
||||||
import { AssetEditActionItem } from 'src/dtos/editing.dto';
|
import { AssetEditActionItem } from 'src/dtos/editing.dto';
|
||||||
import { SourceTypeSchema } from 'src/enum';
|
import { SourceTypeSchema } from 'src/enum';
|
||||||
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
||||||
@@ -40,11 +39,11 @@ const PeopleUpdateSchema = z
|
|||||||
})
|
})
|
||||||
.meta({ id: 'PeopleUpdateDto' });
|
.meta({ id: 'PeopleUpdateDto' });
|
||||||
|
|
||||||
const MergePersonSchema = z
|
const MergeFaceClusterSchema = z
|
||||||
.object({
|
.object({
|
||||||
ids: z.array(z.uuidv4()).describe('Person IDs to merge'),
|
ids: z.array(z.uuidv4()).describe('Face cluster IDs to merge'),
|
||||||
})
|
})
|
||||||
.meta({ id: 'MergePersonDto' });
|
.meta({ id: 'MergeFaceClusterDto' });
|
||||||
|
|
||||||
const PersonSearchSchema = z
|
const PersonSearchSchema = z
|
||||||
.object({
|
.object({
|
||||||
@@ -81,13 +80,14 @@ export const PersonResponseSchema = z
|
|||||||
.optional()
|
.optional()
|
||||||
.describe('Person color (hex)')
|
.describe('Person color (hex)')
|
||||||
.meta(new HistoryBuilder().added('v1.126.0').stable('v2').getExtensions()),
|
.meta(new HistoryBuilder().added('v1.126.0').stable('v2').getExtensions()),
|
||||||
|
faceClusterId: z.string().nullable().describe('Face cluster ID'),
|
||||||
})
|
})
|
||||||
.meta({ id: 'PersonResponseDto' });
|
.meta({ id: 'PersonResponseDto' });
|
||||||
|
|
||||||
export class PersonCreateDto extends createZodDto(PersonCreateSchema) {}
|
export class PersonCreateDto extends createZodDto(PersonCreateSchema) {}
|
||||||
export class PersonUpdateDto extends createZodDto(PersonUpdateSchema) {}
|
export class PersonUpdateDto extends createZodDto(PersonUpdateSchema) {}
|
||||||
export class PeopleUpdateDto extends createZodDto(PeopleUpdateSchema) {}
|
export class PeopleUpdateDto extends createZodDto(PeopleUpdateSchema) {}
|
||||||
export class MergePersonDto extends createZodDto(MergePersonSchema) {}
|
export class MergeFaceClusterDto extends createZodDto(MergeFaceClusterSchema) {}
|
||||||
export class PersonSearchDto extends createZodDto(PersonSearchSchema) {}
|
export class PersonSearchDto extends createZodDto(PersonSearchSchema) {}
|
||||||
export class PersonResponseDto extends createZodDto(PersonResponseSchema) {}
|
export class PersonResponseDto extends createZodDto(PersonResponseSchema) {}
|
||||||
|
|
||||||
@@ -179,6 +179,7 @@ export function mapPerson(person: MaybeDehydrated<Person>): PersonResponseDto {
|
|||||||
isFavorite: person.isFavorite,
|
isFavorite: person.isFavorite,
|
||||||
color: person.color ?? undefined,
|
color: person.color ?? undefined,
|
||||||
updatedAt: asDateString(person.updatedAt),
|
updatedAt: asDateString(person.updatedAt),
|
||||||
|
faceClusterId: person.faceClusterId,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -207,12 +208,11 @@ function mapFacesWithoutPerson(
|
|||||||
|
|
||||||
export function mapFaces(
|
export function mapFaces(
|
||||||
face: AssetFace,
|
face: AssetFace,
|
||||||
auth: AuthDto,
|
|
||||||
edits?: AssetEditActionItem[],
|
edits?: AssetEditActionItem[],
|
||||||
assetDimensions?: ImageDimensions,
|
assetDimensions?: ImageDimensions,
|
||||||
): AssetFaceResponseDto {
|
): AssetFaceResponseDto {
|
||||||
return {
|
return {
|
||||||
...mapFacesWithoutPerson(face, edits, assetDimensions),
|
...mapFacesWithoutPerson(face, edits, assetDimensions),
|
||||||
person: face.person?.ownerId === auth.user.id ? mapPerson(face.person) : null,
|
person: face.person ? mapPerson(face.person) : null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -374,10 +374,13 @@ const SyncAssetFaceV1Schema = z
|
|||||||
})
|
})
|
||||||
.meta({ id: 'SyncAssetFaceV1' });
|
.meta({ id: 'SyncAssetFaceV1' });
|
||||||
|
|
||||||
const SyncAssetFaceV2Schema = SyncAssetFaceV1Schema.extend({
|
const SyncAssetFaceV2Schema = SyncAssetFaceV1Schema.omit({ personId: true })
|
||||||
deletedAt: isoDatetimeToDate.nullable().describe('Face deleted at'),
|
.extend({
|
||||||
isVisible: z.boolean().describe('Is the face visible in the asset'),
|
deletedAt: isoDatetimeToDate.nullable().describe('Face deleted at'),
|
||||||
}).meta({ id: 'SyncAssetFaceV2' });
|
isVisible: z.boolean().describe('Is the face visible in the asset'),
|
||||||
|
faceClusterId: z.string().nullable().describe('Person ID'),
|
||||||
|
})
|
||||||
|
.meta({ id: 'SyncAssetFaceV2' });
|
||||||
|
|
||||||
const SyncAssetFaceDeleteV1Schema = z
|
const SyncAssetFaceDeleteV1Schema = z
|
||||||
.object({ assetFaceId: z.string().describe('Asset face ID') })
|
.object({ assetFaceId: z.string().describe('Asset face ID') })
|
||||||
|
|||||||
@@ -306,6 +306,28 @@ export enum Permission {
|
|||||||
AdminAuthUnlinkAll = 'adminAuth.unlinkAll',
|
AdminAuthUnlinkAll = 'adminAuth.unlinkAll',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export enum SharingPermission {
|
||||||
|
All = 'all',
|
||||||
|
|
||||||
|
AssetRead = 'asset.read',
|
||||||
|
AssetUpdate = 'asset.update',
|
||||||
|
AssetEdit = 'asset.edit',
|
||||||
|
AssetDelete = 'asset.delete',
|
||||||
|
AssetShare = 'asset.share',
|
||||||
|
|
||||||
|
ExifRead = 'exif.read',
|
||||||
|
|
||||||
|
PersonRead = 'person.read',
|
||||||
|
PersonUpdate = 'person.update',
|
||||||
|
PersonMerge = 'person.merge',
|
||||||
|
PersonDelete = 'person.delete',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SharingPermissionSchema = z
|
||||||
|
.enum(SharingPermission)
|
||||||
|
.describe('Sharing permission schema')
|
||||||
|
.meta({ id: 'SharingPermission' });
|
||||||
|
|
||||||
export enum SharedLinkType {
|
export enum SharedLinkType {
|
||||||
Album = 'ALBUM',
|
Album = 'ALBUM',
|
||||||
|
|
||||||
@@ -404,6 +426,7 @@ export enum ManualJobName {
|
|||||||
MemoryCleanup = 'memory-cleanup',
|
MemoryCleanup = 'memory-cleanup',
|
||||||
MemoryCreate = 'memory-create',
|
MemoryCreate = 'memory-create',
|
||||||
BackupDatabase = 'backup-database',
|
BackupDatabase = 'backup-database',
|
||||||
|
PersonGroupMerge = 'person-group-merge',
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ManualJobNameSchema = z.enum(ManualJobName).describe('Manual job name').meta({ id: 'ManualJobName' });
|
export const ManualJobNameSchema = z.enum(ManualJobName).describe('Manual job name').meta({ id: 'ManualJobName' });
|
||||||
@@ -813,6 +836,7 @@ export enum JobName {
|
|||||||
|
|
||||||
FacialRecognitionQueueAll = 'FacialRecognitionQueueAll',
|
FacialRecognitionQueueAll = 'FacialRecognitionQueueAll',
|
||||||
FacialRecognition = 'FacialRecognition',
|
FacialRecognition = 'FacialRecognition',
|
||||||
|
FacialRecognitionMerge = 'FacialRecognitionMerge',
|
||||||
|
|
||||||
FileDelete = 'FileDelete',
|
FileDelete = 'FileDelete',
|
||||||
FileMigrationQueueAll = 'FileMigrationQueueAll',
|
FileMigrationQueueAll = 'FileMigrationQueueAll',
|
||||||
|
|||||||
@@ -149,6 +149,40 @@ where
|
|||||||
"albumAssets"."livePhotoVideoId"
|
"albumAssets"."livePhotoVideoId"
|
||||||
] && array[$2]::uuid[]
|
] && array[$2]::uuid[]
|
||||||
|
|
||||||
|
-- AccessRepository.asset.checkSharedAccess
|
||||||
|
select
|
||||||
|
"album_asset"."assetId"
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_asset"."albumId" = "album_user"."albumId"
|
||||||
|
and "album_user"."userId" = $1
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" in ($2)
|
||||||
|
and "album_asset"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
(
|
||||||
|
"album_user"."permissions" @> $3::sharing_permission_enum[]
|
||||||
|
or $4 = any ("album_user"."permissions")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
union
|
||||||
|
select
|
||||||
|
"asset"."id" as "assetId"
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
inner join "asset" on "asset"."ownerId" = "partner"."sharedById"
|
||||||
|
and "asset"."id" in ($5)
|
||||||
|
where
|
||||||
|
"partner"."sharedWithId" = $6
|
||||||
|
and (
|
||||||
|
"partner"."permissions" @> $7::sharing_permission_enum[]
|
||||||
|
or $8 = any ("partner"."permissions")
|
||||||
|
)
|
||||||
|
|
||||||
-- AccessRepository.authDevice.checkOwnerAccess
|
-- AccessRepository.authDevice.checkOwnerAccess
|
||||||
select
|
select
|
||||||
"session"."id"
|
"session"."id"
|
||||||
|
|||||||
@@ -182,18 +182,25 @@ select
|
|||||||
from
|
from
|
||||||
(
|
(
|
||||||
select
|
select
|
||||||
"asset_face".*,
|
(
|
||||||
"person" as "person"
|
select
|
||||||
|
to_json(obj)
|
||||||
|
from
|
||||||
|
(
|
||||||
|
select
|
||||||
|
"person".*
|
||||||
|
from
|
||||||
|
"face_cluster"
|
||||||
|
inner join "person" on "person"."faceClusterId" = "face_cluster"."id"
|
||||||
|
where
|
||||||
|
"face_cluster"."id" = "asset_face"."faceClusterId"
|
||||||
|
limit
|
||||||
|
$1
|
||||||
|
) as obj
|
||||||
|
) as "person",
|
||||||
|
"asset_face".*
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
left join lateral (
|
|
||||||
select
|
|
||||||
"person".*
|
|
||||||
from
|
|
||||||
"person"
|
|
||||||
where
|
|
||||||
"asset_face"."personId" = "person"."id"
|
|
||||||
) as "person" on true
|
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" = "asset"."id"
|
"asset_face"."assetId" = "asset"."id"
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
@@ -224,7 +231,7 @@ from
|
|||||||
"asset"
|
"asset"
|
||||||
left join "asset_exif" on "asset"."id" = "asset_exif"."assetId"
|
left join "asset_exif" on "asset"."id" = "asset_exif"."assetId"
|
||||||
where
|
where
|
||||||
"asset"."id" = any ($1::uuid[])
|
"asset"."id" = any ($2::uuid[])
|
||||||
|
|
||||||
-- AssetRepository.deleteAll
|
-- AssetRepository.deleteAll
|
||||||
delete from "asset"
|
delete from "asset"
|
||||||
@@ -290,13 +297,44 @@ limit
|
|||||||
|
|
||||||
-- AssetRepository.getById
|
-- AssetRepository.getById
|
||||||
select
|
select
|
||||||
"asset".*
|
"asset".*,
|
||||||
|
(
|
||||||
|
select
|
||||||
|
coalesce(json_agg(agg), '[]')
|
||||||
|
from
|
||||||
|
(
|
||||||
|
select distinct
|
||||||
|
unnest("album_user"."permissions") as "permission"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
inner join "album_asset" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."userId" = "asset"."ownerId"
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = $1
|
||||||
|
)
|
||||||
|
union
|
||||||
|
select distinct
|
||||||
|
unnest("partner"."permissions") as "permission"
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $2
|
||||||
|
) as agg
|
||||||
|
) as "permissions"
|
||||||
from
|
from
|
||||||
"asset"
|
"asset"
|
||||||
where
|
where
|
||||||
"asset"."id" = $1::uuid
|
"asset"."id" = $3::uuid
|
||||||
limit
|
limit
|
||||||
$2
|
$4
|
||||||
|
|
||||||
-- AssetRepository.updateAll
|
-- AssetRepository.updateAll
|
||||||
update "asset"
|
update "asset"
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ select
|
|||||||
$1 as "one"
|
$1 as "one"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
inner join "person" on "person"."id" = "asset_face"."personId"
|
inner join "person" on "person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" = "asset"."id"
|
"asset_face"."assetId" = "asset"."id"
|
||||||
and "person"."isHidden" = $2
|
and "person"."isHidden" = $2
|
||||||
@@ -86,7 +86,7 @@ select
|
|||||||
$1 as "one"
|
$1 as "one"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
inner join "person" on "person"."id" = "asset_face"."personId"
|
inner join "person" on "person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" = "asset"."id"
|
"asset_face"."assetId" = "asset"."id"
|
||||||
and "person"."isHidden" = $2
|
and "person"."isHidden" = $2
|
||||||
|
|||||||
@@ -3,9 +3,6 @@
|
|||||||
-- PersonRepository.reassignFaces
|
-- PersonRepository.reassignFaces
|
||||||
update "asset_face"
|
update "asset_face"
|
||||||
set
|
set
|
||||||
"personId" = $1
|
|
||||||
where
|
|
||||||
"asset_face"."personId" = $2
|
|
||||||
|
|
||||||
-- PersonRepository.delete
|
-- PersonRepository.delete
|
||||||
delete from "person"
|
delete from "person"
|
||||||
@@ -24,27 +21,64 @@ limit
|
|||||||
3
|
3
|
||||||
|
|
||||||
-- PersonRepository.getAllForUser
|
-- PersonRepository.getAllForUser
|
||||||
select
|
select distinct
|
||||||
"person".*
|
on ("person"."faceClusterId") "person".*
|
||||||
from
|
from
|
||||||
"person"
|
"person"
|
||||||
inner join "asset_face" on "asset_face"."personId" = "person"."id"
|
inner join "asset_face" on "asset_face"."faceClusterId" = "person"."faceClusterId"
|
||||||
inner join "asset" on "asset_face"."assetId" = "asset"."id"
|
inner join "asset" on "asset_face"."assetId" = "asset"."id"
|
||||||
and "asset"."visibility" = 'timeline'
|
and "asset"."visibility" = 'timeline'
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
where
|
where
|
||||||
"person"."ownerId" = $1
|
(
|
||||||
|
"person"."ownerId" = $1
|
||||||
|
or (
|
||||||
|
exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "person"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $2
|
||||||
|
and (
|
||||||
|
$3 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $4
|
||||||
|
)
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = $5
|
||||||
|
)
|
||||||
|
and "album_user"."userId" = "person"."ownerId"
|
||||||
|
and (
|
||||||
|
$6 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $7
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" is true
|
and "asset_face"."isVisible" is true
|
||||||
and "person"."isHidden" = $2
|
and "person"."isHidden" = $8
|
||||||
group by
|
group by
|
||||||
"person"."id"
|
"person"."id"
|
||||||
having
|
having
|
||||||
(
|
(
|
||||||
"person"."name" != $3
|
"person"."name" != $9
|
||||||
or count("asset_face"."assetId") >= $4
|
or count("asset_face"."assetId") >= $10
|
||||||
)
|
)
|
||||||
order by
|
order by
|
||||||
|
"person"."faceClusterId",
|
||||||
|
"person"."ownerId" = $11 desc,
|
||||||
"person"."isHidden" asc,
|
"person"."isHidden" asc,
|
||||||
"person"."isFavorite" desc,
|
"person"."isFavorite" desc,
|
||||||
NULLIF(person.name, '') is null asc,
|
NULLIF(person.name, '') is null asc,
|
||||||
@@ -52,16 +86,16 @@ order by
|
|||||||
NULLIF(person.name, '') asc nulls last,
|
NULLIF(person.name, '') asc nulls last,
|
||||||
"person"."createdAt"
|
"person"."createdAt"
|
||||||
limit
|
limit
|
||||||
$5
|
$12
|
||||||
offset
|
offset
|
||||||
$6
|
$13
|
||||||
|
|
||||||
-- PersonRepository.getAllWithoutFaces
|
-- PersonRepository.getAllWithoutFaces
|
||||||
select
|
select
|
||||||
"person".*
|
"person".*
|
||||||
from
|
from
|
||||||
"person"
|
"person"
|
||||||
left join "asset_face" on "asset_face"."personId" = "person"."id"
|
left join "asset_face" on "asset_face"."faceClusterId" = "person"."faceClusterId"
|
||||||
where
|
where
|
||||||
"asset_face"."deletedAt" is null
|
"asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" is true
|
and "asset_face"."isVisible" is true
|
||||||
@@ -83,15 +117,26 @@ select
|
|||||||
from
|
from
|
||||||
"person"
|
"person"
|
||||||
where
|
where
|
||||||
"person"."id" = "asset_face"."personId"
|
"person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
|
order by
|
||||||
|
"person"."ownerId" = (
|
||||||
|
select
|
||||||
|
"asset"."ownerId"
|
||||||
|
from
|
||||||
|
"asset"
|
||||||
|
where
|
||||||
|
"asset"."id" = "asset_face"."assetId"
|
||||||
|
) desc
|
||||||
|
limit
|
||||||
|
$1
|
||||||
) as obj
|
) as obj
|
||||||
) as "person"
|
) as "person"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" = $1
|
"asset_face"."assetId" = $2
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" = $2
|
and "asset_face"."isVisible" = $3
|
||||||
order by
|
order by
|
||||||
"asset_face"."boundingBoxX1" asc
|
"asset_face"."boundingBoxX1" asc
|
||||||
|
|
||||||
@@ -108,19 +153,30 @@ select
|
|||||||
from
|
from
|
||||||
"person"
|
"person"
|
||||||
where
|
where
|
||||||
"person"."id" = "asset_face"."personId"
|
"person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
|
order by
|
||||||
|
"person"."ownerId" = (
|
||||||
|
select
|
||||||
|
"asset"."ownerId"
|
||||||
|
from
|
||||||
|
"asset"
|
||||||
|
where
|
||||||
|
"asset"."id" = "asset_face"."assetId"
|
||||||
|
) desc
|
||||||
|
limit
|
||||||
|
$1
|
||||||
) as obj
|
) as obj
|
||||||
) as "person"
|
) as "person"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
where
|
where
|
||||||
"asset_face"."id" = $1
|
"asset_face"."id" = $2
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
|
|
||||||
-- PersonRepository.getFaceForFacialRecognitionJob
|
-- PersonRepository.getFaceForFacialRecognitionJob
|
||||||
select
|
select
|
||||||
"asset_face"."id",
|
"asset_face"."id",
|
||||||
"asset_face"."personId",
|
"asset_face"."faceClusterId",
|
||||||
"asset_face"."sourceType",
|
"asset_face"."sourceType",
|
||||||
(
|
(
|
||||||
select
|
select
|
||||||
@@ -190,7 +246,7 @@ where
|
|||||||
-- PersonRepository.reassignFace
|
-- PersonRepository.reassignFace
|
||||||
update "asset_face"
|
update "asset_face"
|
||||||
set
|
set
|
||||||
"personId" = $1
|
"faceClusterId" = $1
|
||||||
where
|
where
|
||||||
"asset_face"."id" = $2
|
"asset_face"."id" = $2
|
||||||
|
|
||||||
@@ -209,9 +265,10 @@ where
|
|||||||
"person"."ownerId" = $1
|
"person"."ownerId" = $1
|
||||||
and f_unaccent ("person"."name") %> f_unaccent ($2)
|
and f_unaccent ("person"."name") %> f_unaccent ($2)
|
||||||
order by
|
order by
|
||||||
f_unaccent ("person"."name") <->>> f_unaccent ($3)
|
f_unaccent ("person"."name") <->>> f_unaccent ($3),
|
||||||
|
"person"."ownerId" = $4 desc
|
||||||
limit
|
limit
|
||||||
$4
|
$5
|
||||||
|
|
||||||
-- PersonRepository.getDistinctNames
|
-- PersonRepository.getDistinctNames
|
||||||
select distinct
|
select distinct
|
||||||
@@ -234,9 +291,52 @@ from
|
|||||||
and "asset"."visibility" = 'timeline'
|
and "asset"."visibility" = 'timeline'
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
where
|
where
|
||||||
"asset_face"."deletedAt" is null
|
(
|
||||||
|
"asset"."ownerId" = $1
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $2
|
||||||
|
and (
|
||||||
|
$3 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $4
|
||||||
|
)
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $5
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$6 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $7
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" is true
|
and "asset_face"."isVisible" is true
|
||||||
and "asset_face"."personId" = $1
|
and "asset_face"."faceClusterId" = (
|
||||||
|
select
|
||||||
|
"person"."faceClusterId"
|
||||||
|
from
|
||||||
|
"person"
|
||||||
|
where
|
||||||
|
"person"."id" = $8
|
||||||
|
)
|
||||||
|
|
||||||
-- PersonRepository.getNumberOfPeople
|
-- PersonRepository.getNumberOfPeople
|
||||||
select
|
select
|
||||||
@@ -256,7 +356,7 @@ where
|
|||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
where
|
where
|
||||||
"asset_face"."personId" = "person"."id"
|
"asset_face"."faceClusterId" = "person"."faceClusterId"
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" = $2
|
and "asset_face"."isVisible" = $2
|
||||||
and exists (
|
and exists (
|
||||||
@@ -269,7 +369,42 @@ where
|
|||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
and "person"."ownerId" = $3
|
and (
|
||||||
|
"person"."ownerId" = $3
|
||||||
|
or (
|
||||||
|
exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "person"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $4
|
||||||
|
and (
|
||||||
|
$5 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $6
|
||||||
|
)
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = $7
|
||||||
|
)
|
||||||
|
and "album_user"."userId" = "person"."ownerId"
|
||||||
|
and (
|
||||||
|
$8 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $9
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
-- PersonRepository.refreshFaces
|
-- PersonRepository.refreshFaces
|
||||||
with
|
with
|
||||||
@@ -299,14 +434,26 @@ select
|
|||||||
from
|
from
|
||||||
"person"
|
"person"
|
||||||
where
|
where
|
||||||
"person"."id" = "asset_face"."personId"
|
"person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
|
order by
|
||||||
|
"person"."ownerId" = (
|
||||||
|
select
|
||||||
|
"asset"."ownerId"
|
||||||
|
from
|
||||||
|
"asset"
|
||||||
|
where
|
||||||
|
"asset"."id" = "asset_face"."assetId"
|
||||||
|
) desc
|
||||||
|
limit
|
||||||
|
$1
|
||||||
) as obj
|
) as obj
|
||||||
) as "person"
|
) as "person"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
|
inner join "person" on "person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" in ($1)
|
"person"."id" in ($2)
|
||||||
and "asset_face"."personId" in ($2)
|
and "asset_face"."assetId" in ($3)
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
|
|
||||||
-- PersonRepository.getRandomFace
|
-- PersonRepository.getRandomFace
|
||||||
@@ -314,8 +461,52 @@ select
|
|||||||
"asset_face".*
|
"asset_face".*
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
|
inner join "person" on "asset_face"."faceClusterId" = "person"."faceClusterId"
|
||||||
|
and "person"."id" = $1
|
||||||
where
|
where
|
||||||
"asset_face"."personId" = $1
|
"asset_face"."assetId" in (
|
||||||
|
select
|
||||||
|
"asset"."id"
|
||||||
|
from
|
||||||
|
"asset"
|
||||||
|
where
|
||||||
|
(
|
||||||
|
"asset"."ownerId" = "person"."ownerId"
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = "person"."ownerId"
|
||||||
|
and (
|
||||||
|
$2 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $3
|
||||||
|
)
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = "person"."ownerId"
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$4 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $5
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
and "asset_face"."deletedAt" is null
|
and "asset_face"."deletedAt" is null
|
||||||
and "asset_face"."isVisible" is true
|
and "asset_face"."isVisible" is true
|
||||||
|
|
||||||
@@ -351,8 +542,9 @@ select
|
|||||||
"asset_face"."id"
|
"asset_face"."id"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
|
inner join "person" on "person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
|
and "person"."id" = $1
|
||||||
inner join "asset" on "asset"."id" = "asset_face"."assetId"
|
inner join "asset" on "asset"."id" = "asset_face"."assetId"
|
||||||
and "asset"."isOffline" = $1
|
and "asset"."isOffline" = $2
|
||||||
where
|
where
|
||||||
"asset_face"."assetId" = $2
|
"asset_face"."assetId" = $3
|
||||||
and "asset_face"."personId" = $3
|
|
||||||
|
|||||||
@@ -10,15 +10,52 @@ where
|
|||||||
"asset"."visibility" = $1
|
"asset"."visibility" = $1
|
||||||
and "asset"."fileCreatedAt" >= $2
|
and "asset"."fileCreatedAt" >= $2
|
||||||
and "asset_exif"."lensModel" = $3
|
and "asset_exif"."lensModel" = $3
|
||||||
and "asset"."ownerId" = any ($4::uuid[])
|
and (
|
||||||
and "asset"."isFavorite" = $5
|
"asset"."ownerId" = $4
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $5
|
||||||
|
and (
|
||||||
|
$6 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $7
|
||||||
|
)
|
||||||
|
and "partner"."inTimeline" = $8
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $9
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."inTimeline" = $10
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$11 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $12
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset"."isFavorite" = $13
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
order by
|
order by
|
||||||
"asset"."fileCreatedAt" desc
|
"asset"."fileCreatedAt" desc
|
||||||
limit
|
limit
|
||||||
$6
|
$14
|
||||||
offset
|
offset
|
||||||
$7
|
$15
|
||||||
|
|
||||||
-- SearchRepository.searchStatistics
|
-- SearchRepository.searchStatistics
|
||||||
select
|
select
|
||||||
@@ -30,8 +67,45 @@ where
|
|||||||
"asset"."visibility" = $1
|
"asset"."visibility" = $1
|
||||||
and "asset"."fileCreatedAt" >= $2
|
and "asset"."fileCreatedAt" >= $2
|
||||||
and "asset_exif"."lensModel" = $3
|
and "asset_exif"."lensModel" = $3
|
||||||
and "asset"."ownerId" = any ($4::uuid[])
|
and (
|
||||||
and "asset"."isFavorite" = $5
|
"asset"."ownerId" = $4
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $5
|
||||||
|
and (
|
||||||
|
$6 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $7
|
||||||
|
)
|
||||||
|
and "partner"."inTimeline" = $8
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $9
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."inTimeline" = $10
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$11 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $12
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset"."isFavorite" = $13
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
|
|
||||||
-- SearchRepository.searchRandom
|
-- SearchRepository.searchRandom
|
||||||
@@ -44,13 +118,50 @@ where
|
|||||||
"asset"."visibility" = $1
|
"asset"."visibility" = $1
|
||||||
and "asset"."fileCreatedAt" >= $2
|
and "asset"."fileCreatedAt" >= $2
|
||||||
and "asset_exif"."lensModel" = $3
|
and "asset_exif"."lensModel" = $3
|
||||||
and "asset"."ownerId" = any ($4::uuid[])
|
and (
|
||||||
and "asset"."isFavorite" = $5
|
"asset"."ownerId" = $4
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $5
|
||||||
|
and (
|
||||||
|
$6 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $7
|
||||||
|
)
|
||||||
|
and "partner"."inTimeline" = $8
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $9
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."inTimeline" = $10
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$11 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $12
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset"."isFavorite" = $13
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
order by
|
order by
|
||||||
random()
|
random()
|
||||||
limit
|
limit
|
||||||
$6
|
$14
|
||||||
|
|
||||||
-- SearchRepository.searchLargeAssets
|
-- SearchRepository.searchLargeAssets
|
||||||
select
|
select
|
||||||
@@ -63,14 +174,51 @@ where
|
|||||||
"asset"."visibility" = $1
|
"asset"."visibility" = $1
|
||||||
and "asset"."fileCreatedAt" >= $2
|
and "asset"."fileCreatedAt" >= $2
|
||||||
and "asset_exif"."lensModel" = $3
|
and "asset_exif"."lensModel" = $3
|
||||||
and "asset"."ownerId" = any ($4::uuid[])
|
and (
|
||||||
and "asset"."isFavorite" = $5
|
"asset"."ownerId" = $4
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $5
|
||||||
|
and (
|
||||||
|
$6 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $7
|
||||||
|
)
|
||||||
|
and "partner"."inTimeline" = $8
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $9
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."inTimeline" = $10
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$11 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $12
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset"."isFavorite" = $13
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
and "asset_exif"."fileSizeInByte" > $6
|
and "asset_exif"."fileSizeInByte" > $14
|
||||||
order by
|
order by
|
||||||
"asset_exif"."fileSizeInByte" desc
|
"asset_exif"."fileSizeInByte" desc
|
||||||
limit
|
limit
|
||||||
$7
|
$15
|
||||||
|
|
||||||
-- SearchRepository.searchSmart
|
-- SearchRepository.searchSmart
|
||||||
begin
|
begin
|
||||||
@@ -86,15 +234,52 @@ where
|
|||||||
"asset"."visibility" = $1
|
"asset"."visibility" = $1
|
||||||
and "asset"."fileCreatedAt" >= $2
|
and "asset"."fileCreatedAt" >= $2
|
||||||
and "asset_exif"."lensModel" = $3
|
and "asset_exif"."lensModel" = $3
|
||||||
and "asset"."ownerId" = any ($4::uuid[])
|
and (
|
||||||
and "asset"."isFavorite" = $5
|
"asset"."ownerId" = $4
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"partner"."sharedById" = "asset"."ownerId"
|
||||||
|
and "partner"."sharedWithId" = $5
|
||||||
|
and (
|
||||||
|
$6 = any ("partner"."permissions")
|
||||||
|
or "partner"."permissions" @> $7
|
||||||
|
)
|
||||||
|
and "partner"."inTimeline" = $8
|
||||||
|
)
|
||||||
|
or exists (
|
||||||
|
select
|
||||||
|
from
|
||||||
|
"album_asset"
|
||||||
|
inner join "album_user" on "album_user"."albumId" = "album_asset"."albumId"
|
||||||
|
and "album_user"."userId" = $9
|
||||||
|
where
|
||||||
|
"album_asset"."assetId" = "asset"."id"
|
||||||
|
and "album_user"."inTimeline" = $10
|
||||||
|
and "album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = "asset"."ownerId"
|
||||||
|
and (
|
||||||
|
$11 = any ("album_user"."permissions")
|
||||||
|
or "album_user"."permissions" @> $12
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
and "asset"."isFavorite" = $13
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
order by
|
order by
|
||||||
smart_search.embedding <=> $6
|
smart_search.embedding <=> $14
|
||||||
limit
|
limit
|
||||||
$7
|
$15
|
||||||
offset
|
offset
|
||||||
$8
|
$16
|
||||||
commit
|
commit
|
||||||
|
|
||||||
-- SearchRepository.getEmbedding
|
-- SearchRepository.getEmbedding
|
||||||
@@ -113,15 +298,30 @@ with
|
|||||||
"cte" as (
|
"cte" as (
|
||||||
select
|
select
|
||||||
"asset_face"."id",
|
"asset_face"."id",
|
||||||
"asset_face"."personId",
|
"asset_face"."faceClusterId",
|
||||||
face_search.embedding <=> $1 as "distance"
|
face_search.embedding <=> $1 as "distance",
|
||||||
|
"asset"."ownerId"
|
||||||
from
|
from
|
||||||
"asset_face"
|
"asset_face"
|
||||||
inner join "asset" on "asset"."id" = "asset_face"."assetId"
|
inner join "asset" on "asset"."id" = "asset_face"."assetId"
|
||||||
inner join "face_search" on "face_search"."faceId" = "asset_face"."id"
|
inner join "face_search" on "face_search"."faceId" = "asset_face"."id"
|
||||||
left join "person" on "person"."id" = "asset_face"."personId"
|
left join "person" on "person"."faceClusterId" = "asset_face"."faceClusterId"
|
||||||
where
|
where
|
||||||
"asset"."ownerId" = any ($2::uuid[])
|
"asset"."ownerId" in (
|
||||||
|
select
|
||||||
|
"user"."id"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."trustedGroupId" in (
|
||||||
|
select
|
||||||
|
"user"."trustedGroupId"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."id" = any ($2::uuid[])
|
||||||
|
)
|
||||||
|
)
|
||||||
and "asset"."deletedAt" is null
|
and "asset"."deletedAt" is null
|
||||||
order by
|
order by
|
||||||
"distance"
|
"distance"
|
||||||
|
|||||||
@@ -527,7 +527,7 @@ order by
|
|||||||
select
|
select
|
||||||
"asset_face"."id",
|
"asset_face"."id",
|
||||||
"assetId",
|
"assetId",
|
||||||
"personId",
|
"faceClusterId",
|
||||||
"imageWidth",
|
"imageWidth",
|
||||||
"imageHeight",
|
"imageHeight",
|
||||||
"boundingBoxX1",
|
"boundingBoxX1",
|
||||||
|
|||||||
@@ -397,3 +397,73 @@ set
|
|||||||
where
|
where
|
||||||
"user"."deletedAt" is null
|
"user"."deletedAt" is null
|
||||||
and "user"."id" = $2::uuid
|
and "user"."id" = $2::uuid
|
||||||
|
|
||||||
|
-- UserRepository.getInSameTrustedGroup
|
||||||
|
select
|
||||||
|
"user"."id"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."trustedGroupId" = (
|
||||||
|
select
|
||||||
|
"user"."trustedGroupId"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."id" = $1
|
||||||
|
)
|
||||||
|
|
||||||
|
-- UserRepository.mergeTrustedGroups
|
||||||
|
update "user"
|
||||||
|
set
|
||||||
|
"trustedGroupId" = "u"."trustedGroupId"
|
||||||
|
from
|
||||||
|
"user" as "u"
|
||||||
|
where
|
||||||
|
"u"."id" = $1
|
||||||
|
and "user"."trustedGroupId" = (
|
||||||
|
select
|
||||||
|
"user"."trustedGroupId"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."id" = $2
|
||||||
|
and "user"."trustedGroupId" != "u"."trustedGroupId"
|
||||||
|
)
|
||||||
|
|
||||||
|
-- UserRepository.updateTrustedGroups
|
||||||
|
update "user"
|
||||||
|
set
|
||||||
|
"trustedGroupId" = uuid_generate_v4 ()
|
||||||
|
where
|
||||||
|
"user"."trustedGroupId" = (
|
||||||
|
select
|
||||||
|
"user"."trustedGroupId"
|
||||||
|
from
|
||||||
|
"user"
|
||||||
|
where
|
||||||
|
"user"."id" = $1
|
||||||
|
)
|
||||||
|
and "user"."id" != $2
|
||||||
|
and "user"."id" not in (
|
||||||
|
select
|
||||||
|
"partner"."sharedById" as "userId"
|
||||||
|
from
|
||||||
|
"partner"
|
||||||
|
where
|
||||||
|
"sharedWithId" = $3
|
||||||
|
union
|
||||||
|
select
|
||||||
|
"album_user"."userId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."albumId" in (
|
||||||
|
select
|
||||||
|
"album_user"."albumId"
|
||||||
|
from
|
||||||
|
"album_user"
|
||||||
|
where
|
||||||
|
"album_user"."userId" = $4
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|||||||
@@ -2,7 +2,9 @@ import { Injectable } from '@nestjs/common';
|
|||||||
import { Kysely, NotNull, sql } from 'kysely';
|
import { Kysely, NotNull, sql } from 'kysely';
|
||||||
import { InjectKysely } from 'nestjs-kysely';
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
import { ChunkedSet, DummyValue, GenerateSql } from 'src/decorators';
|
import { ChunkedSet, DummyValue, GenerateSql } from 'src/decorators';
|
||||||
import { AlbumUserRole, AssetVisibility } from 'src/enum';
|
import { AlbumUserRole, AssetVisibility, SharingPermission } from 'src/enum';
|
||||||
|
import { hasAssetPermissions } from 'src/repositories/asset.repository';
|
||||||
|
import { hasPermissions } from 'src/repositories/person.repository';
|
||||||
import { DB } from 'src/schema';
|
import { DB } from 'src/schema';
|
||||||
import { asUuid } from 'src/utils/database';
|
import { asUuid } from 'src/utils/database';
|
||||||
|
|
||||||
@@ -273,6 +275,46 @@ class AssetAccess {
|
|||||||
return allowedIds;
|
return allowedIds;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.UUID_SET, [SharingPermission.All]] })
|
||||||
|
async checkSharedAccess(userId: string, assetIds: Set<string>, permissions: SharingPermission[]) {
|
||||||
|
const ids = await this.db
|
||||||
|
.selectFrom('album_asset')
|
||||||
|
.select('album_asset.assetId')
|
||||||
|
.where('album_asset.assetId', 'in', [...assetIds])
|
||||||
|
.where('album_asset.albumId', 'in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select('album_user.albumId')
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb('album_user.permissions', '@>', sql<SharingPermission[]>`${permissions}::sharing_permission_enum[]`),
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('album_user.permissions')),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.innerJoin('album_user', (join) =>
|
||||||
|
join.onRef('album_asset.albumId', '=', 'album_user.albumId').on('album_user.userId', '=', userId),
|
||||||
|
)
|
||||||
|
.union((eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.where('partner.sharedWithId', '=', userId)
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb('partner.permissions', '@>', sql<SharingPermission[]>`${permissions}::sharing_permission_enum[]`),
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('partner.permissions')),
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
.innerJoin('asset', (join) =>
|
||||||
|
join.onRef('asset.ownerId', '=', 'partner.sharedById').on('asset.id', 'in', [...assetIds]),
|
||||||
|
)
|
||||||
|
.select('asset.id as assetId'),
|
||||||
|
)
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return new Set(ids.map(({ assetId }) => assetId));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class AuthDeviceAccess {
|
class AuthDeviceAccess {
|
||||||
@@ -452,6 +494,37 @@ class PersonAccess {
|
|||||||
.execute()
|
.execute()
|
||||||
.then((faces) => new Set(faces.map((face) => face.id)));
|
.then((faces) => new Set(faces.map((face) => face.id)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async checkSharedAccess(userId: string, personIds: Set<string>, permissions: SharingPermission[]) {
|
||||||
|
if (personIds.size === 0) {
|
||||||
|
return new Set<string>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const ids = await this.db
|
||||||
|
.selectFrom('person')
|
||||||
|
.select('person.id')
|
||||||
|
.where('person.id', 'in', [...personIds])
|
||||||
|
.where(hasPermissions(userId, permissions))
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return new Set(ids.map(({ id }) => id));
|
||||||
|
}
|
||||||
|
|
||||||
|
async checkSharedFaceAccess(userId: string, faceIds: Set<string>, permissions: SharingPermission[]) {
|
||||||
|
if (faceIds.size === 0) {
|
||||||
|
return new Set<string>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const ids = await this.db
|
||||||
|
.selectFrom('asset_face')
|
||||||
|
.select('asset_face.id')
|
||||||
|
.leftJoin('asset', (join) => join.onRef('asset.id', '=', 'asset_face.assetId'))
|
||||||
|
.where('asset_face.id', 'in', [...faceIds])
|
||||||
|
.where(hasAssetPermissions(userId, permissions))
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return new Set(ids.map(({ id }) => id));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class PartnerAccess {
|
class PartnerAccess {
|
||||||
|
|||||||
@@ -38,4 +38,13 @@ export class AlbumUserRepository {
|
|||||||
async delete({ userId, albumId }: AlbumPermissionId): Promise<void> {
|
async delete({ userId, albumId }: AlbumPermissionId): Promise<void> {
|
||||||
await this.db.deleteFrom('album_user').where('userId', '=', userId).where('albumId', '=', albumId).execute();
|
await this.db.deleteFrom('album_user').where('userId', '=', userId).where('albumId', '=', albumId).execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get({ userId, albumId }: AlbumPermissionId) {
|
||||||
|
return this.db
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select(['permissions', 'inTimeline'])
|
||||||
|
.where('userId', '=', userId)
|
||||||
|
.where('albumId', '=', albumId)
|
||||||
|
.executeTakeFirstOrThrow();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
SelectQueryBuilder,
|
SelectQueryBuilder,
|
||||||
ShallowDehydrateObject,
|
ShallowDehydrateObject,
|
||||||
sql,
|
sql,
|
||||||
|
StringReference,
|
||||||
Updateable,
|
Updateable,
|
||||||
UpdateResult,
|
UpdateResult,
|
||||||
} from 'kysely';
|
} from 'kysely';
|
||||||
@@ -17,7 +18,15 @@ import { InjectKysely } from 'nestjs-kysely';
|
|||||||
import { LockableProperty, Stack } from 'src/database';
|
import { LockableProperty, Stack } from 'src/database';
|
||||||
import { Chunked, ChunkedArray, DummyValue, GenerateSql } from 'src/decorators';
|
import { Chunked, ChunkedArray, DummyValue, GenerateSql } from 'src/decorators';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import { AssetFileType, AssetOrder, AssetOrderBy, AssetStatus, AssetType, AssetVisibility } from 'src/enum';
|
import {
|
||||||
|
AssetFileType,
|
||||||
|
AssetOrder,
|
||||||
|
AssetOrderBy,
|
||||||
|
AssetStatus,
|
||||||
|
AssetType,
|
||||||
|
AssetVisibility,
|
||||||
|
SharingPermission,
|
||||||
|
} from 'src/enum';
|
||||||
import { DB } from 'src/schema';
|
import { DB } from 'src/schema';
|
||||||
import { AssetAudioTable, AssetKeyframeTable, AssetVideoTable } from 'src/schema/tables/asset-av.table';
|
import { AssetAudioTable, AssetKeyframeTable, AssetVideoTable } from 'src/schema/tables/asset-av.table';
|
||||||
import { AssetExifTable } from 'src/schema/tables/asset-exif.table';
|
import { AssetExifTable } from 'src/schema/tables/asset-exif.table';
|
||||||
@@ -41,6 +50,7 @@ import {
|
|||||||
withFiles,
|
withFiles,
|
||||||
withLibrary,
|
withLibrary,
|
||||||
withOwner,
|
withOwner,
|
||||||
|
withPermissions,
|
||||||
withSmartSearch,
|
withSmartSearch,
|
||||||
withTagId,
|
withTagId,
|
||||||
withTags,
|
withTags,
|
||||||
@@ -165,6 +175,93 @@ const withBoundingBox = <T>(qb: SelectQueryBuilder<DB, 'asset' | 'asset_exif', T
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const hasAssetPermissions =
|
||||||
|
(userId: string, permissions: SharingPermission[], ignoreTimelineVisibility: boolean = false) =>
|
||||||
|
(eb: ExpressionBuilder<DB, 'asset'>) =>
|
||||||
|
eb.or([
|
||||||
|
eb('asset.ownerId', '=', userId),
|
||||||
|
eb.exists(
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.whereRef('partner.sharedById', '=', 'asset.ownerId')
|
||||||
|
.where('partner.sharedWithId', '=', userId)
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('partner.permissions')),
|
||||||
|
eb('partner.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
.$if(!ignoreTimelineVisibility, (qb) => qb.where('partner.inTimeline', '=', true)),
|
||||||
|
),
|
||||||
|
eb.exists(
|
||||||
|
eb
|
||||||
|
.selectFrom('album_asset')
|
||||||
|
.whereRef('album_asset.assetId', '=', 'asset.id')
|
||||||
|
.innerJoin('album_user', (join) =>
|
||||||
|
join.onRef('album_user.albumId', '=', 'album_asset.albumId').on('album_user.userId', '=', userId),
|
||||||
|
)
|
||||||
|
.$if(!ignoreTimelineVisibility, (qb) => qb.where('album_user.inTimeline', '=', true))
|
||||||
|
.where('album_user.albumId', 'in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select('album_user.albumId')
|
||||||
|
.whereRef('album_user.userId', '=', 'asset.ownerId')
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('album_user.permissions')),
|
||||||
|
eb('album_user.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
|
||||||
|
export const hasAssetPermissionsRef = <T extends keyof DB>(
|
||||||
|
eb: ExpressionBuilder<DB, 'asset'>,
|
||||||
|
userIdRef: StringReference<DB, 'asset' | T>,
|
||||||
|
permissions: SharingPermission[],
|
||||||
|
ignoreTimelineVisibility: boolean = false,
|
||||||
|
) =>
|
||||||
|
eb.or([
|
||||||
|
eb('asset.ownerId', '=', eb.ref(userIdRef as never)),
|
||||||
|
eb.exists(
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.whereRef('partner.sharedById', '=', 'asset.ownerId')
|
||||||
|
.whereRef('partner.sharedWithId', '=', userIdRef as never)
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('partner.permissions')),
|
||||||
|
eb('partner.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
.$if(!ignoreTimelineVisibility, (qb) => qb.where('partner.inTimeline', '=', true)),
|
||||||
|
),
|
||||||
|
eb.exists(
|
||||||
|
eb
|
||||||
|
.selectFrom('album_asset')
|
||||||
|
.whereRef('album_asset.assetId', '=', 'asset.id')
|
||||||
|
.innerJoin('album_user', (join) =>
|
||||||
|
join
|
||||||
|
.onRef('album_user.albumId', '=', 'album_asset.albumId')
|
||||||
|
.onRef('album_user.userId', '=', userIdRef as never),
|
||||||
|
)
|
||||||
|
.$if(!ignoreTimelineVisibility, (qb) => qb.where('album_user.inTimeline', '=', true))
|
||||||
|
.where('album_user.albumId', 'in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select('album_user.albumId')
|
||||||
|
.whereRef('album_user.userId', '=', 'asset.ownerId')
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('album_user.permissions')),
|
||||||
|
eb('album_user.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class AssetRepository {
|
export class AssetRepository {
|
||||||
constructor(@InjectKysely() private db: Kysely<DB>) {}
|
constructor(@InjectKysely() private db: Kysely<DB>) {}
|
||||||
@@ -556,17 +653,22 @@ export class AssetRepository {
|
|||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID] })
|
@GenerateSql({ params: [DummyValue.UUID, {}, DummyValue.UUID] })
|
||||||
getById(
|
getById(
|
||||||
id: string,
|
id: string,
|
||||||
{ exifInfo, faces, files, library, owner, smartSearch, stack, tags, edits }: GetByIdsRelations = {},
|
{ exifInfo, faces, files, library, owner, smartSearch, stack, tags, edits }: GetByIdsRelations = {},
|
||||||
|
userId?: string,
|
||||||
) {
|
) {
|
||||||
return this.db
|
return this.db
|
||||||
.selectFrom('asset')
|
.selectFrom('asset')
|
||||||
.selectAll('asset')
|
.selectAll('asset')
|
||||||
.where('asset.id', '=', asUuid(id))
|
.where('asset.id', '=', asUuid(id))
|
||||||
.$if(!!exifInfo, withExif)
|
.$if(!!exifInfo, withExif)
|
||||||
.$if(!!faces, (qb) => qb.select(faces?.person ? withFacesAndPeople : withFaces).$narrowType<{ faces: NotNull }>())
|
.$if(!!faces, (qb) =>
|
||||||
|
qb
|
||||||
|
.select(faces?.person ? (eb) => withFacesAndPeople(eb, { userId }) : withFaces)
|
||||||
|
.$narrowType<{ faces: NotNull }>(),
|
||||||
|
)
|
||||||
.$if(!!library, (qb) => qb.select(withLibrary))
|
.$if(!!library, (qb) => qb.select(withLibrary))
|
||||||
.$if(!!owner, (qb) => qb.select(withOwner))
|
.$if(!!owner, (qb) => qb.select(withOwner))
|
||||||
.$if(!!smartSearch, withSmartSearch)
|
.$if(!!smartSearch, withSmartSearch)
|
||||||
@@ -602,6 +704,7 @@ export class AssetRepository {
|
|||||||
.$if(!!files, (qb) => qb.select(withFiles))
|
.$if(!!files, (qb) => qb.select(withFiles))
|
||||||
.$if(!!tags, (qb) => qb.select(withTags))
|
.$if(!!tags, (qb) => qb.select(withTags))
|
||||||
.$if(!!edits, (qb) => qb.select(withEdits))
|
.$if(!!edits, (qb) => qb.select(withEdits))
|
||||||
|
.$if(!!userId, (qb) => qb.select(withPermissions(userId!)))
|
||||||
.limit(1)
|
.limit(1)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
@@ -744,7 +847,9 @@ export class AssetRepository {
|
|||||||
)
|
)
|
||||||
.where((eb) => eb.or([eb('asset.stackId', 'is', null), eb(eb.table('stack'), 'is not', null)])),
|
.where((eb) => eb.or([eb('asset.stackId', 'is', null), eb(eb.table('stack'), 'is not', null)])),
|
||||||
)
|
)
|
||||||
.$if(!!options.userIds, (qb) => qb.where('asset.ownerId', '=', anyUuid(options.userIds!)))
|
.$if(!!options.userIds, (qb) =>
|
||||||
|
qb.where(hasAssetPermissions(options.userIds![0], [SharingPermission.AssetRead], !!options.personId)),
|
||||||
|
)
|
||||||
.$if(options.isFavorite !== undefined, (qb) => qb.where('asset.isFavorite', '=', options.isFavorite!))
|
.$if(options.isFavorite !== undefined, (qb) => qb.where('asset.isFavorite', '=', options.isFavorite!))
|
||||||
.$if(!!options.assetType, (qb) => qb.where('asset.type', '=', options.assetType!))
|
.$if(!!options.assetType, (qb) => qb.where('asset.type', '=', options.assetType!))
|
||||||
.$if(options.isDuplicate !== undefined, (qb) =>
|
.$if(options.isDuplicate !== undefined, (qb) =>
|
||||||
@@ -829,7 +934,9 @@ export class AssetRepository {
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
.$if(!!options.personId, (qb) => hasPeople(qb, [options.personId!]))
|
.$if(!!options.personId, (qb) => hasPeople(qb, [options.personId!]))
|
||||||
.$if(!!options.userIds, (qb) => qb.where('asset.ownerId', '=', anyUuid(options.userIds!)))
|
.$if(!!options.userIds, (qb) =>
|
||||||
|
qb.where(hasAssetPermissions(options.userIds![0], [SharingPermission.AssetRead], !!options.personId)),
|
||||||
|
)
|
||||||
.$if(options.isFavorite !== undefined, (qb) => qb.where('asset.isFavorite', '=', options.isFavorite!))
|
.$if(options.isFavorite !== undefined, (qb) => qb.where('asset.isFavorite', '=', options.isFavorite!))
|
||||||
.$if(!!options.withStacked, (qb) =>
|
.$if(!!options.withStacked, (qb) =>
|
||||||
qb
|
qb
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ import { getKeyByValue, getMethodNames, ImmichStartupError } from 'src/utils/mis
|
|||||||
type JobMapItem = {
|
type JobMapItem = {
|
||||||
jobName: JobName;
|
jobName: JobName;
|
||||||
queueName: QueueName;
|
queueName: QueueName;
|
||||||
handler: (job: JobOf<any>) => Promise<JobStatus>;
|
handler: (job?: JobOf<any>) => Promise<JobStatus>;
|
||||||
label: string;
|
label: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -95,14 +95,17 @@ export class JobRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async run({ name, data }: JobItem) {
|
async run(job: JobItem) {
|
||||||
const item = this.handlers[name as JobName];
|
const item = this.handlers[job.name];
|
||||||
if (!item) {
|
if (!item) {
|
||||||
this.logger.warn(`Skipping unknown job: "${name}"`);
|
this.logger.warn(`Skipping unknown job: "${job.name}"`);
|
||||||
return JobStatus.Skipped;
|
return JobStatus.Skipped;
|
||||||
}
|
}
|
||||||
|
|
||||||
return item.handler(data);
|
if ('data' in job) {
|
||||||
|
return item.handler(job.data);
|
||||||
|
}
|
||||||
|
return item.handler();
|
||||||
}
|
}
|
||||||
|
|
||||||
setConcurrency(queueName: QueueName, concurrency: number) {
|
setConcurrency(queueName: QueueName, concurrency: number) {
|
||||||
@@ -167,7 +170,7 @@ export class JobRepository {
|
|||||||
const queueName = this.getQueueName(item.name);
|
const queueName = this.getQueueName(item.name);
|
||||||
const job = {
|
const job = {
|
||||||
name: item.name,
|
name: item.name,
|
||||||
data: item.data || {},
|
data: ('data' in item ? item.data : undefined) || {},
|
||||||
options: this.getJobOptions(item) || undefined,
|
options: this.getJobOptions(item) || undefined,
|
||||||
} as JobItem & { data: any; options: JobsOptions | undefined };
|
} as JobItem & { data: any; options: JobsOptions | undefined };
|
||||||
|
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ export class MemoryRepository implements IBulkAsset {
|
|||||||
eb.exists(
|
eb.exists(
|
||||||
eb
|
eb
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.innerJoin('person', 'person.id', 'asset_face.personId')
|
.innerJoin('person', 'person.faceClusterId', 'asset_face.faceClusterId')
|
||||||
.select((eb) => eb.val(1).as('one'))
|
.select((eb) => eb.val(1).as('one'))
|
||||||
.whereRef('asset_face.assetId', '=', 'asset.id')
|
.whereRef('asset_face.assetId', '=', 'asset.id')
|
||||||
.where('person.isHidden', '=', true),
|
.where('person.isHidden', '=', true),
|
||||||
|
|||||||
@@ -4,7 +4,8 @@ import { jsonObjectFrom } from 'kysely/helpers/postgres';
|
|||||||
import { InjectKysely } from 'nestjs-kysely';
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
import { AssetFace } from 'src/database';
|
import { AssetFace } from 'src/database';
|
||||||
import { Chunked, ChunkedArray, DummyValue, GenerateSql } from 'src/decorators';
|
import { Chunked, ChunkedArray, DummyValue, GenerateSql } from 'src/decorators';
|
||||||
import { AssetFileType, AssetVisibility, SourceType } from 'src/enum';
|
import { AssetFileType, AssetVisibility, SharingPermission, SourceType } from 'src/enum';
|
||||||
|
import { hasAssetPermissions, hasAssetPermissionsRef } from 'src/repositories/asset.repository';
|
||||||
import { DB } from 'src/schema';
|
import { DB } from 'src/schema';
|
||||||
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
||||||
import { FaceSearchTable } from 'src/schema/tables/face-search.table';
|
import { FaceSearchTable } from 'src/schema/tables/face-search.table';
|
||||||
@@ -33,9 +34,9 @@ export interface AssetFaceId {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface UpdateFacesData {
|
export interface UpdateFacesData {
|
||||||
oldPersonId?: string;
|
oldFaceClusterId?: string;
|
||||||
faceIds?: string[];
|
faceIds?: string[];
|
||||||
newPersonId: string;
|
newFaceClusterId: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PersonStatistics {
|
export interface PersonStatistics {
|
||||||
@@ -54,7 +55,7 @@ export interface GetAllPeopleOptions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface GetAllFacesOptions {
|
export interface GetAllFacesOptions {
|
||||||
personId?: string | null;
|
faceClusterId?: string | null;
|
||||||
assetId?: string;
|
assetId?: string;
|
||||||
sourceType?: SourceType;
|
sourceType?: SourceType;
|
||||||
}
|
}
|
||||||
@@ -63,9 +64,27 @@ export type UnassignFacesOptions = DeleteFacesOptions;
|
|||||||
|
|
||||||
export type SelectFaceOptions = (keyof Selectable<AssetFaceTable>)[];
|
export type SelectFaceOptions = (keyof Selectable<AssetFaceTable>)[];
|
||||||
|
|
||||||
const withPerson = (eb: ExpressionBuilder<DB, 'asset_face'>) => {
|
const withPerson = (eb: ExpressionBuilder<DB, 'asset_face'>, userId?: string) => {
|
||||||
return jsonObjectFrom(
|
return jsonObjectFrom(
|
||||||
eb.selectFrom('person').selectAll('person').whereRef('person.id', '=', 'asset_face.personId'),
|
eb
|
||||||
|
.selectFrom('person')
|
||||||
|
.selectAll('person')
|
||||||
|
.whereRef('person.faceClusterId', '=', 'asset_face.faceClusterId')
|
||||||
|
.$if(!!userId, (qb) =>
|
||||||
|
qb.where((eb) =>
|
||||||
|
eb.or([eb('person.ownerId', '=', userId!), hasPermissions(userId!, [SharingPermission.PersonRead])(eb)]),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.orderBy(
|
||||||
|
(eb) =>
|
||||||
|
eb(
|
||||||
|
'person.ownerId',
|
||||||
|
'=',
|
||||||
|
eb.selectFrom('asset').select('asset.ownerId').whereRef('asset.id', '=', 'asset_face.assetId'),
|
||||||
|
),
|
||||||
|
'desc',
|
||||||
|
)
|
||||||
|
.limit(1),
|
||||||
).as('person');
|
).as('person');
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -75,16 +94,47 @@ const withFaceSearch = (eb: ExpressionBuilder<DB, 'asset_face'>) => {
|
|||||||
).as('faceSearch');
|
).as('faceSearch');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const hasPermissions =
|
||||||
|
(userId: string, permissions: SharingPermission[]) => (eb: ExpressionBuilder<DB, 'person'>) =>
|
||||||
|
eb.or([
|
||||||
|
eb.exists((eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.whereRef('partner.sharedById', '=', 'person.ownerId')
|
||||||
|
.where('partner.sharedWithId', '=', userId)
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('partner.permissions')),
|
||||||
|
eb('partner.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
eb.exists((eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.where('album_user.albumId', 'in', (eb) =>
|
||||||
|
eb.selectFrom('album_user').select('album_user.albumId').where('album_user.userId', '=', userId),
|
||||||
|
)
|
||||||
|
.whereRef('album_user.userId', '=', 'person.ownerId')
|
||||||
|
.where((eb) =>
|
||||||
|
eb.or([
|
||||||
|
eb(eb.val(SharingPermission.All), '=', eb.fn.any('album_user.permissions')),
|
||||||
|
eb('album_user.permissions', '@>', eb.val(permissions)),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class PersonRepository {
|
export class PersonRepository {
|
||||||
constructor(@InjectKysely() private db: Kysely<DB>) {}
|
constructor(@InjectKysely() private db: Kysely<DB>) {}
|
||||||
|
|
||||||
@GenerateSql({ params: [{ oldPersonId: DummyValue.UUID, newPersonId: DummyValue.UUID }] })
|
@GenerateSql({ params: [{ oldPersonId: DummyValue.UUID, newPersonId: DummyValue.UUID }] })
|
||||||
async reassignFaces({ oldPersonId, faceIds, newPersonId }: UpdateFacesData): Promise<number> {
|
async reassignFaces({ oldFaceClusterId, faceIds, newFaceClusterId }: UpdateFacesData): Promise<number> {
|
||||||
const result = await this.db
|
const result = await this.db
|
||||||
.updateTable('asset_face')
|
.updateTable('asset_face')
|
||||||
.set({ personId: newPersonId })
|
.set({ faceClusterId: newFaceClusterId })
|
||||||
.$if(!!oldPersonId, (qb) => qb.where('asset_face.personId', '=', oldPersonId!))
|
.$if(!!oldFaceClusterId, (qb) => qb.where('asset_face.faceClusterId', '=', oldFaceClusterId!))
|
||||||
.$if(!!faceIds, (qb) => qb.where('asset_face.id', 'in', faceIds!))
|
.$if(!!faceIds, (qb) => qb.where('asset_face.id', 'in', faceIds!))
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
|
|
||||||
@@ -94,7 +144,7 @@ export class PersonRepository {
|
|||||||
async unassignFaces({ sourceType }: UnassignFacesOptions): Promise<void> {
|
async unassignFaces({ sourceType }: UnassignFacesOptions): Promise<void> {
|
||||||
await this.db
|
await this.db
|
||||||
.updateTable('asset_face')
|
.updateTable('asset_face')
|
||||||
.set({ personId: null })
|
.set({ faceClusterId: null })
|
||||||
.where('asset_face.sourceType', '=', sourceType)
|
.where('asset_face.sourceType', '=', sourceType)
|
||||||
.execute();
|
.execute();
|
||||||
}
|
}
|
||||||
@@ -117,8 +167,8 @@ export class PersonRepository {
|
|||||||
return this.db
|
return this.db
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.selectAll('asset_face')
|
.selectAll('asset_face')
|
||||||
.$if(options.personId === null, (qb) => qb.where('asset_face.personId', 'is', null))
|
.$if(options.faceClusterId === null, (qb) => qb.where('asset_face.faceClusterId', 'is', null))
|
||||||
.$if(!!options.personId, (qb) => qb.where('asset_face.personId', '=', options.personId!))
|
.$if(!!options.faceClusterId, (qb) => qb.where('asset_face.faceClusterId', '=', options.faceClusterId!))
|
||||||
.$if(!!options.sourceType, (qb) => qb.where('asset_face.sourceType', '=', options.sourceType!))
|
.$if(!!options.sourceType, (qb) => qb.where('asset_face.sourceType', '=', options.sourceType!))
|
||||||
.$if(!!options.assetId, (qb) => qb.where('asset_face.assetId', '=', options.assetId!))
|
.$if(!!options.assetId, (qb) => qb.where('asset_face.assetId', '=', options.assetId!))
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
@@ -153,16 +203,20 @@ export class PersonRepository {
|
|||||||
const items = await this.db
|
const items = await this.db
|
||||||
.selectFrom('person')
|
.selectFrom('person')
|
||||||
.selectAll('person')
|
.selectAll('person')
|
||||||
.innerJoin('asset_face', 'asset_face.personId', 'person.id')
|
.innerJoin('asset_face', 'asset_face.faceClusterId', 'person.faceClusterId')
|
||||||
.innerJoin('asset', (join) =>
|
.innerJoin('asset', (join) =>
|
||||||
join
|
join
|
||||||
.onRef('asset_face.assetId', '=', 'asset.id')
|
.onRef('asset_face.assetId', '=', 'asset.id')
|
||||||
.on('asset.visibility', '=', sql.lit(AssetVisibility.Timeline))
|
.on('asset.visibility', '=', sql.lit(AssetVisibility.Timeline))
|
||||||
.on('asset.deletedAt', 'is', null),
|
.on('asset.deletedAt', 'is', null),
|
||||||
)
|
)
|
||||||
.where('person.ownerId', '=', userId)
|
.where((eb) =>
|
||||||
|
eb.or([eb('person.ownerId', '=', userId), hasPermissions(userId, [SharingPermission.PersonRead])(eb)]),
|
||||||
|
)
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.where('asset_face.isVisible', 'is', true)
|
.where('asset_face.isVisible', 'is', true)
|
||||||
|
.orderBy('person.faceClusterId')
|
||||||
|
.orderBy((eb) => eb('person.ownerId', '=', userId), 'desc')
|
||||||
.orderBy('person.isHidden', 'asc')
|
.orderBy('person.isHidden', 'asc')
|
||||||
.orderBy('person.isFavorite', 'desc')
|
.orderBy('person.isFavorite', 'desc')
|
||||||
.having((eb) =>
|
.having((eb) =>
|
||||||
@@ -171,6 +225,7 @@ export class PersonRepository {
|
|||||||
eb((innerEb) => innerEb.fn.count('asset_face.assetId'), '>=', options?.minimumFaceCount || 1),
|
eb((innerEb) => innerEb.fn.count('asset_face.assetId'), '>=', options?.minimumFaceCount || 1),
|
||||||
]),
|
]),
|
||||||
)
|
)
|
||||||
|
.distinctOn('person.faceClusterId')
|
||||||
.groupBy('person.id')
|
.groupBy('person.id')
|
||||||
.$if(!!options?.closestFaceAssetId, (qb) =>
|
.$if(!!options?.closestFaceAssetId, (qb) =>
|
||||||
qb.orderBy((eb) =>
|
qb.orderBy((eb) =>
|
||||||
@@ -209,7 +264,7 @@ export class PersonRepository {
|
|||||||
return this.db
|
return this.db
|
||||||
.selectFrom('person')
|
.selectFrom('person')
|
||||||
.selectAll('person')
|
.selectAll('person')
|
||||||
.leftJoin('asset_face', 'asset_face.personId', 'person.id')
|
.leftJoin('asset_face', 'asset_face.faceClusterId', 'person.faceClusterId')
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.where('asset_face.isVisible', 'is', true)
|
.where('asset_face.isVisible', 'is', true)
|
||||||
.having((eb) => eb.fn.count('asset_face.assetId'), '=', 0)
|
.having((eb) => eb.fn.count('asset_face.assetId'), '=', 0)
|
||||||
@@ -218,13 +273,13 @@ export class PersonRepository {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID] })
|
@GenerateSql({ params: [DummyValue.UUID] })
|
||||||
getFaces(assetId: string, options?: { isVisible?: boolean }) {
|
getFaces(assetId: string, options: { isVisible?: boolean; userId?: string } = {}) {
|
||||||
const isVisible = options === undefined ? true : options.isVisible;
|
const { isVisible = true, userId } = options;
|
||||||
|
|
||||||
return this.db
|
return this.db
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.selectAll('asset_face')
|
.selectAll('asset_face')
|
||||||
.select(withPerson)
|
.select((eb) => withPerson(eb, userId))
|
||||||
.where('asset_face.assetId', '=', assetId)
|
.where('asset_face.assetId', '=', assetId)
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.$if(isVisible !== undefined, (qb) => qb.where('asset_face.isVisible', '=', isVisible!))
|
.$if(isVisible !== undefined, (qb) => qb.where('asset_face.isVisible', '=', isVisible!))
|
||||||
@@ -248,7 +303,7 @@ export class PersonRepository {
|
|||||||
getFaceForFacialRecognitionJob(id: string) {
|
getFaceForFacialRecognitionJob(id: string) {
|
||||||
return this.db
|
return this.db
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.select(['asset_face.id', 'asset_face.personId', 'asset_face.sourceType'])
|
.select(['asset_face.id', 'asset_face.faceClusterId', 'asset_face.sourceType'])
|
||||||
.select((eb) =>
|
.select((eb) =>
|
||||||
jsonObjectFrom(
|
jsonObjectFrom(
|
||||||
eb
|
eb
|
||||||
@@ -289,10 +344,10 @@ export class PersonRepository {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID, DummyValue.UUID] })
|
@GenerateSql({ params: [DummyValue.UUID, DummyValue.UUID] })
|
||||||
async reassignFace(assetFaceId: string, newPersonId: string): Promise<number> {
|
async reassignFace(assetFaceId: string, newFaceClusterId: string): Promise<number> {
|
||||||
const result = await this.db
|
const result = await this.db
|
||||||
.updateTable('asset_face')
|
.updateTable('asset_face')
|
||||||
.set({ personId: newPersonId })
|
.set({ faceClusterId: newFaceClusterId })
|
||||||
.where('asset_face.id', '=', assetFaceId)
|
.where('asset_face.id', '=', assetFaceId)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
|
|
||||||
@@ -318,6 +373,7 @@ export class PersonRepository {
|
|||||||
.where('person.ownerId', '=', userId)
|
.where('person.ownerId', '=', userId)
|
||||||
.where(() => sql`f_unaccent("person"."name") %> f_unaccent(${personName})`)
|
.where(() => sql`f_unaccent("person"."name") %> f_unaccent(${personName})`)
|
||||||
.orderBy(sql`f_unaccent("person"."name") <->>> f_unaccent(${personName})`)
|
.orderBy(sql`f_unaccent("person"."name") <->>> f_unaccent(${personName})`)
|
||||||
|
.orderBy((eb) => eb('person.ownerId', '=', userId), 'desc')
|
||||||
.limit(100)
|
.limit(100)
|
||||||
.$if(!withHidden, (qb) => qb.where('person.isHidden', '=', false))
|
.$if(!withHidden, (qb) => qb.where('person.isHidden', '=', false))
|
||||||
.execute();
|
.execute();
|
||||||
@@ -335,7 +391,7 @@ export class PersonRepository {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@GenerateSql({ params: [DummyValue.UUID] })
|
@GenerateSql({ params: [DummyValue.UUID] })
|
||||||
async getStatistics(personId: string): Promise<PersonStatistics> {
|
async getStatistics(userId: string, personId: string): Promise<PersonStatistics> {
|
||||||
const result = await this.db
|
const result = await this.db
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.leftJoin('asset', (join) =>
|
.leftJoin('asset', (join) =>
|
||||||
@@ -344,10 +400,13 @@ export class PersonRepository {
|
|||||||
.on('asset.visibility', '=', sql.lit(AssetVisibility.Timeline))
|
.on('asset.visibility', '=', sql.lit(AssetVisibility.Timeline))
|
||||||
.on('asset.deletedAt', 'is', null),
|
.on('asset.deletedAt', 'is', null),
|
||||||
)
|
)
|
||||||
|
.where(hasAssetPermissions(userId, [SharingPermission.AssetRead], true))
|
||||||
.select((eb) => eb.fn.count(eb.fn('distinct', ['asset.id'])).as('count'))
|
.select((eb) => eb.fn.count(eb.fn('distinct', ['asset.id'])).as('count'))
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.where('asset_face.isVisible', 'is', true)
|
.where('asset_face.isVisible', 'is', true)
|
||||||
.where('asset_face.personId', '=', personId)
|
.where('asset_face.faceClusterId', '=', (eb) =>
|
||||||
|
eb.selectFrom('person').select('person.faceClusterId').where('person.id', '=', personId),
|
||||||
|
)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@@ -364,7 +423,7 @@ export class PersonRepository {
|
|||||||
eb.exists((eb) =>
|
eb.exists((eb) =>
|
||||||
eb
|
eb
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.whereRef('asset_face.personId', '=', 'person.id')
|
.whereRef('asset_face.faceClusterId', '=', 'person.faceClusterId')
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.where('asset_face.isVisible', '=', true)
|
.where('asset_face.isVisible', '=', true)
|
||||||
.where((eb) =>
|
.where((eb) =>
|
||||||
@@ -378,13 +437,20 @@ export class PersonRepository {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
.where('person.ownerId', '=', userId)
|
.where((eb) =>
|
||||||
|
eb.or([eb('person.ownerId', '=', userId), hasPermissions(userId, [SharingPermission.PersonRead])(eb)]),
|
||||||
|
)
|
||||||
.select((eb) => eb.fn.coalesce(eb.fn.countAll<number>(), zero).as('total'))
|
.select((eb) => eb.fn.coalesce(eb.fn.countAll<number>(), zero).as('total'))
|
||||||
.select((eb) => eb.fn.coalesce(eb.fn.countAll<number>().filterWhere('isHidden', '=', true), zero).as('hidden'))
|
.select((eb) => eb.fn.coalesce(eb.fn.countAll<number>().filterWhere('isHidden', '=', true), zero).as('hidden'))
|
||||||
.executeTakeFirstOrThrow();
|
.executeTakeFirstOrThrow();
|
||||||
}
|
}
|
||||||
|
|
||||||
create(person: Insertable<PersonTable>) {
|
async create(person: Insertable<PersonTable>) {
|
||||||
|
if (!person.faceClusterId) {
|
||||||
|
const { id } = await this.db.insertInto('face_cluster').defaultValues().returning('id').executeTakeFirstOrThrow();
|
||||||
|
person.faceClusterId = id;
|
||||||
|
}
|
||||||
|
|
||||||
return this.db.insertInto('person').values(person).returningAll().executeTakeFirstOrThrow();
|
return this.db.insertInto('person').values(person).returningAll().executeTakeFirstOrThrow();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -475,8 +541,9 @@ export class PersonRepository {
|
|||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.selectAll('asset_face')
|
.selectAll('asset_face')
|
||||||
.select(withPerson)
|
.select(withPerson)
|
||||||
|
.innerJoin('person', (join) => join.onRef('person.faceClusterId', '=', 'asset_face.faceClusterId'))
|
||||||
|
.where('person.id', 'in', personIds)
|
||||||
.where('asset_face.assetId', 'in', assetIds)
|
.where('asset_face.assetId', 'in', assetIds)
|
||||||
.where('asset_face.personId', 'in', personIds)
|
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.execute();
|
.execute();
|
||||||
}
|
}
|
||||||
@@ -486,7 +553,15 @@ export class PersonRepository {
|
|||||||
return this.db
|
return this.db
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.selectAll('asset_face')
|
.selectAll('asset_face')
|
||||||
.where('asset_face.personId', '=', personId)
|
.innerJoin('person', (join) =>
|
||||||
|
join.onRef('asset_face.faceClusterId', '=', 'person.faceClusterId').on('person.id', '=', personId),
|
||||||
|
)
|
||||||
|
.where('asset_face.assetId', 'in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('asset')
|
||||||
|
.select('asset.id')
|
||||||
|
.where((eb) => hasAssetPermissionsRef(eb, 'person.ownerId', [SharingPermission.AssetRead], true)),
|
||||||
|
)
|
||||||
.where('asset_face.deletedAt', 'is', null)
|
.where('asset_face.deletedAt', 'is', null)
|
||||||
.where('asset_face.isVisible', 'is', true)
|
.where('asset_face.isVisible', 'is', true)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
@@ -573,8 +648,14 @@ export class PersonRepository {
|
|||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.select('asset_face.id')
|
.select('asset_face.id')
|
||||||
.where('asset_face.assetId', '=', assetId)
|
.where('asset_face.assetId', '=', assetId)
|
||||||
.where('asset_face.personId', '=', personId)
|
.innerJoin('person', (join) =>
|
||||||
|
join.onRef('person.faceClusterId', '=', 'asset_face.faceClusterId').on('person.id', '=', personId),
|
||||||
|
)
|
||||||
.innerJoin('asset', (join) => join.onRef('asset.id', '=', 'asset_face.assetId').on('asset.isOffline', '=', false))
|
.innerJoin('asset', (join) => join.onRef('asset.id', '=', 'asset_face.assetId').on('asset.isOffline', '=', false))
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getByFaceClusterId(faceClusterId: string) {
|
||||||
|
return this.db.selectFrom('person').selectAll().where('person.faceClusterId', '=', faceClusterId).execute();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -325,15 +325,23 @@ export class SearchRepository {
|
|||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.select([
|
.select([
|
||||||
'asset_face.id',
|
'asset_face.id',
|
||||||
'asset_face.personId',
|
'asset_face.faceClusterId',
|
||||||
sql<number>`face_search.embedding <=> ${embedding}`.as('distance'),
|
sql<number>`face_search.embedding <=> ${embedding}`.as('distance'),
|
||||||
])
|
])
|
||||||
.innerJoin('asset', 'asset.id', 'asset_face.assetId')
|
.innerJoin('asset', 'asset.id', 'asset_face.assetId')
|
||||||
|
.select('asset.ownerId')
|
||||||
.innerJoin('face_search', 'face_search.faceId', 'asset_face.id')
|
.innerJoin('face_search', 'face_search.faceId', 'asset_face.id')
|
||||||
.leftJoin('person', 'person.id', 'asset_face.personId')
|
.leftJoin('person', 'person.faceClusterId', 'asset_face.faceClusterId')
|
||||||
.where('asset.ownerId', '=', anyUuid(userIds))
|
.where('asset.ownerId', 'in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('user')
|
||||||
|
.select('user.id')
|
||||||
|
.where('user.trustedGroupId', 'in', (eb) =>
|
||||||
|
eb.selectFrom('user').select('user.trustedGroupId').where('user.id', '=', anyUuid(userIds)),
|
||||||
|
),
|
||||||
|
)
|
||||||
.where('asset.deletedAt', 'is', null)
|
.where('asset.deletedAt', 'is', null)
|
||||||
.$if(!!hasPerson, (qb) => qb.where('asset_face.personId', 'is not', null))
|
.$if(!!hasPerson, (qb) => qb.where('asset_face.faceClusterId', 'is not', null))
|
||||||
.$if(!!minBirthDate, (qb) =>
|
.$if(!!minBirthDate, (qb) =>
|
||||||
qb.where((eb) =>
|
qb.where((eb) =>
|
||||||
eb.or([eb('person.birthDate', 'is', null), eb('person.birthDate', '<=', minBirthDate!)]),
|
eb.or([eb('person.birthDate', 'is', null), eb('person.birthDate', '<=', minBirthDate!)]),
|
||||||
|
|||||||
@@ -443,7 +443,7 @@ class AssetFaceSync extends BaseSync {
|
|||||||
.select([
|
.select([
|
||||||
'asset_face.id',
|
'asset_face.id',
|
||||||
'assetId',
|
'assetId',
|
||||||
'personId',
|
'faceClusterId',
|
||||||
'imageWidth',
|
'imageWidth',
|
||||||
'imageHeight',
|
'imageHeight',
|
||||||
'boundingBoxX1',
|
'boundingBoxX1',
|
||||||
|
|||||||
@@ -325,4 +325,61 @@ export class UserRepository {
|
|||||||
|
|
||||||
await query.execute();
|
await query.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@GenerateSql({ params: [DummyValue.UUID] })
|
||||||
|
async getInSameTrustedGroup(userId: string) {
|
||||||
|
return this.db
|
||||||
|
.selectFrom('user')
|
||||||
|
.select('user.id')
|
||||||
|
.where('user.trustedGroupId', '=', (eb) =>
|
||||||
|
eb.selectFrom('user').select('user.trustedGroupId').where('user.id', '=', userId),
|
||||||
|
)
|
||||||
|
.execute()
|
||||||
|
.then((result) => result.map(({ id }) => id));
|
||||||
|
}
|
||||||
|
|
||||||
|
@GenerateSql({ params: [{ userId: DummyValue.UUID, userIdToMerge: DummyValue.UUID }] })
|
||||||
|
async mergeTrustedGroups({ userId, userIdToMerge }: { userId: string; userIdToMerge: string }) {
|
||||||
|
return this.db
|
||||||
|
.updateTable('user')
|
||||||
|
.from('user as u')
|
||||||
|
.where('u.id', '=', userId)
|
||||||
|
.where('user.trustedGroupId', '=', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('user')
|
||||||
|
.select('user.trustedGroupId')
|
||||||
|
.where('user.id', '=', userIdToMerge)
|
||||||
|
.whereRef('user.trustedGroupId', '!=', 'u.trustedGroupId'),
|
||||||
|
)
|
||||||
|
.set((eb) => ({
|
||||||
|
trustedGroupId: eb.ref('u.trustedGroupId'),
|
||||||
|
}))
|
||||||
|
.executeTakeFirst();
|
||||||
|
}
|
||||||
|
|
||||||
|
@GenerateSql({ params: [DummyValue.UUID] })
|
||||||
|
async updateTrustedGroups(userId: string) {
|
||||||
|
return this.db
|
||||||
|
.updateTable('user')
|
||||||
|
.set((eb) => ({ trustedGroupId: eb.fn('uuid_generate_v4') }))
|
||||||
|
.where('user.trustedGroupId', '=', (eb) =>
|
||||||
|
eb.selectFrom('user').select('user.trustedGroupId').where('user.id', '=', userId),
|
||||||
|
)
|
||||||
|
.where('user.id', '!=', userId)
|
||||||
|
.where('user.id', 'not in', (eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.select('partner.sharedById as userId')
|
||||||
|
.where('sharedWithId', '=', userId)
|
||||||
|
.union((eb) =>
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select('album_user.userId')
|
||||||
|
.where('album_user.albumId', 'in', (eb) =>
|
||||||
|
eb.selectFrom('album_user').select('album_user.albumId').where('album_user.userId', '=', userId),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.executeTakeFirst();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import {
|
|||||||
AssetStatus,
|
AssetStatus,
|
||||||
AssetVisibility,
|
AssetVisibility,
|
||||||
ChecksumAlgorithm,
|
ChecksumAlgorithm,
|
||||||
|
SharingPermission,
|
||||||
SourceType,
|
SourceType,
|
||||||
VideoSegmentCodec,
|
VideoSegmentCodec,
|
||||||
} from 'src/enum';
|
} from 'src/enum';
|
||||||
@@ -37,3 +38,8 @@ export const video_stream_variant_codec_enum = registerEnum({
|
|||||||
name: 'video_stream_variant_codec_enum',
|
name: 'video_stream_variant_codec_enum',
|
||||||
values: Object.values(VideoSegmentCodec),
|
values: Object.values(VideoSegmentCodec),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const sharing_permission_enum = registerEnum({
|
||||||
|
name: 'sharing_permission_enum',
|
||||||
|
values: Object.values(SharingPermission),
|
||||||
|
});
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import {
|
|||||||
asset_face_source_type,
|
asset_face_source_type,
|
||||||
asset_visibility_enum,
|
asset_visibility_enum,
|
||||||
assets_status_enum,
|
assets_status_enum,
|
||||||
|
sharing_permission_enum,
|
||||||
} from 'src/schema/enums';
|
} from 'src/schema/enums';
|
||||||
import {
|
import {
|
||||||
album_user_after_insert,
|
album_user_after_insert,
|
||||||
@@ -45,6 +46,7 @@ import { AssetMetadataAuditTable } from 'src/schema/tables/asset-metadata-audit.
|
|||||||
import { AssetMetadataTable } from 'src/schema/tables/asset-metadata.table';
|
import { AssetMetadataTable } from 'src/schema/tables/asset-metadata.table';
|
||||||
import { AssetOcrTable } from 'src/schema/tables/asset-ocr.table';
|
import { AssetOcrTable } from 'src/schema/tables/asset-ocr.table';
|
||||||
import { AssetTable } from 'src/schema/tables/asset.table';
|
import { AssetTable } from 'src/schema/tables/asset.table';
|
||||||
|
import { FaceClusterTable } from 'src/schema/tables/face-cluster.table';
|
||||||
import { FaceSearchTable } from 'src/schema/tables/face-search.table';
|
import { FaceSearchTable } from 'src/schema/tables/face-search.table';
|
||||||
import { GeodataPlacesTable } from 'src/schema/tables/geodata-places.table';
|
import { GeodataPlacesTable } from 'src/schema/tables/geodata-places.table';
|
||||||
import { LibraryTable } from 'src/schema/tables/library.table';
|
import { LibraryTable } from 'src/schema/tables/library.table';
|
||||||
@@ -110,6 +112,7 @@ export class ImmichDatabase {
|
|||||||
AssetTable,
|
AssetTable,
|
||||||
AssetFileTable,
|
AssetFileTable,
|
||||||
AssetExifTable,
|
AssetExifTable,
|
||||||
|
FaceClusterTable,
|
||||||
FaceSearchTable,
|
FaceSearchTable,
|
||||||
GeodataPlacesTable,
|
GeodataPlacesTable,
|
||||||
LibraryTable,
|
LibraryTable,
|
||||||
@@ -170,7 +173,13 @@ export class ImmichDatabase {
|
|||||||
asset_face_audit,
|
asset_face_audit,
|
||||||
];
|
];
|
||||||
|
|
||||||
enum = [album_user_role_enum, assets_status_enum, asset_face_source_type, asset_visibility_enum];
|
enum = [
|
||||||
|
album_user_role_enum,
|
||||||
|
assets_status_enum,
|
||||||
|
asset_face_source_type,
|
||||||
|
asset_visibility_enum,
|
||||||
|
sharing_permission_enum,
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Migrations {
|
export interface Migrations {
|
||||||
@@ -211,6 +220,7 @@ export interface DB {
|
|||||||
ocr_search: OcrSearchTable;
|
ocr_search: OcrSearchTable;
|
||||||
|
|
||||||
face_search: FaceSearchTable;
|
face_search: FaceSearchTable;
|
||||||
|
face_cluster: FaceClusterTable;
|
||||||
|
|
||||||
geodata_places: GeodataPlacesTable;
|
geodata_places: GeodataPlacesTable;
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
import { Kysely, sql } from 'kysely';
|
|
||||||
|
|
||||||
export async function up(db: Kysely<any>): Promise<void> {
|
|
||||||
await sql`ALTER TABLE "workflow" ADD "updateId" uuid NOT NULL DEFAULT immich_uuid_v7();`.execute(db);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function down(db: Kysely<any>): Promise<void> {
|
|
||||||
await sql`ALTER TABLE "workflow" DROP COLUMN "updateId";`.execute(db);
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import { Kysely, sql } from 'kysely';
|
||||||
|
|
||||||
|
export async function up(db: Kysely<any>): Promise<void> {
|
||||||
|
await sql`CREATE TYPE "sharing_permission_enum" AS ENUM ('all','asset.read','asset.update','asset.edit','asset.delete','asset.share','exif.read','person.read','person.update','person.merge','person.delete');`.execute(db);
|
||||||
|
await sql`ALTER TABLE "user" ADD "trustedGroupId" uuid NOT NULL DEFAULT uuid_generate_v4();`.execute(db);
|
||||||
|
await sql`ALTER TABLE "album_user" ADD "permissions" sharing_permission_enum[] NOT NULL DEFAULT '{asset.read,exif.read}';`.execute(db);
|
||||||
|
await sql`ALTER TABLE "album_user" ADD "inTimeline" boolean NOT NULL DEFAULT false;`.execute(db);
|
||||||
|
await sql`ALTER TABLE "partner" ADD "permissions" sharing_permission_enum[] NOT NULL DEFAULT '{all}';`.execute(db);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function down(db: Kysely<any>): Promise<void> {
|
||||||
|
await sql`DROP TYPE "sharing_permission_enum";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "partner" DROP COLUMN "permissions";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "user" DROP COLUMN "trustedGroupId";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "album_user" DROP COLUMN "permissions";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "album_user" DROP COLUMN "inTimeline";`.execute(db);
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
import { Kysely, sql } from 'kysely';
|
||||||
|
|
||||||
|
export async function up(db: Kysely<any>): Promise<void> {
|
||||||
|
await sql`ALTER TABLE "asset_face" RENAME COLUMN "personId" TO "faceClusterId";`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_faceClusterId_assetId_idx" ON "asset_face" ("faceClusterId", "assetId");`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_faceClusterId_assetId_notDeleted_isVisible_idx" ON "asset_face" ("faceClusterId", "assetId") WHERE ("deletedAt" IS NULL AND "isVisible" IS TRUE);`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_assetId_faceClusterId_idx" ON "asset_face" ("assetId", "faceClusterId");`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_personId_assetId_notDeleted_isVisible_idx";`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_assetId_personId_idx";`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_personId_assetId_idx";`.execute(db);
|
||||||
|
await sql`CREATE TABLE "face_cluster" (
|
||||||
|
"id" uuid NOT NULL DEFAULT uuid_generate_v4(),
|
||||||
|
"createdAt" timestamp with time zone NOT NULL DEFAULT now(),
|
||||||
|
"updatedAt" timestamp with time zone NOT NULL DEFAULT now(),
|
||||||
|
"updateId" uuid NOT NULL DEFAULT immich_uuid_v7(),
|
||||||
|
CONSTRAINT "face_cluster_pkey" PRIMARY KEY ("id")
|
||||||
|
);`.execute(db);
|
||||||
|
await sql`ALTER TABLE "asset_face" ADD CONSTRAINT "asset_face_faceClusterId_fkey" FOREIGN KEY ("faceClusterId") REFERENCES "face_cluster" ("id") ON UPDATE CASCADE ON DELETE SET NULL;`.execute(db);
|
||||||
|
await sql`ALTER TABLE "asset_face" DROP CONSTRAINT "asset_face_personId_fkey";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "person" ADD "faceClusterId" uuid;`.execute(db);
|
||||||
|
await sql`CREATE INDEX "person_faceClusterId_idx" ON "person" ("faceClusterId");`.execute(db);
|
||||||
|
await sql`ALTER TABLE "person" ADD CONSTRAINT "person_faceClusterId_fkey" FOREIGN KEY ("faceClusterId") REFERENCES "face_cluster" ("id") ON UPDATE CASCADE ON DELETE CASCADE;`.execute(db);
|
||||||
|
await sql`CREATE INDEX "face_cluster_updateId_idx" ON "face_cluster" ("updateId");`.execute(db);
|
||||||
|
await sql`CREATE OR REPLACE TRIGGER "face_cluster_updatedAt"
|
||||||
|
BEFORE UPDATE ON "face_cluster"
|
||||||
|
FOR EACH ROW
|
||||||
|
EXECUTE FUNCTION updated_at();`.execute(db);
|
||||||
|
await sql`INSERT INTO "migration_overrides" ("name", "value") VALUES ('trigger_face_cluster_updatedAt', '{"type":"trigger","name":"face_cluster_updatedAt","sql":"CREATE OR REPLACE TRIGGER \\"face_cluster_updatedAt\\"\\n BEFORE UPDATE ON \\"face_cluster\\"\\n FOR EACH ROW\\n EXECUTE FUNCTION updated_at();"}'::jsonb);`.execute(db);
|
||||||
|
await sql`INSERT INTO "migration_overrides" ("name", "value") VALUES ('index_asset_face_faceClusterId_assetId_notDeleted_isVisible_idx', '{"type":"index","name":"asset_face_faceClusterId_assetId_notDeleted_isVisible_idx","sql":"CREATE INDEX \\"asset_face_faceClusterId_assetId_notDeleted_isVisible_idx\\" ON \\"asset_face\\" (\\"faceClusterId\\", \\"assetId\\") WHERE (\\"deletedAt\\" IS NULL AND \\"isVisible\\" IS TRUE);"}'::jsonb);`.execute(db);
|
||||||
|
await sql`DELETE FROM "migration_overrides" WHERE "name" = 'index_asset_face_personId_assetId_notDeleted_isVisible_idx';`.execute(db);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function down(db: Kysely<any>): Promise<void> {
|
||||||
|
await sql`ALTER TABLE "person" DROP COLUMN "faceClusterId";`.execute(db);
|
||||||
|
await sql`DROP INDEX "person_faceClusterId_idx";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "person" DROP CONSTRAINT "person_faceClusterId_fkey";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "asset_face" RENAME COLUMN "faceClusterId" TO "personId";`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_personId_assetId_notDeleted_isVisible_idx" ON "asset_face" ("personId", "assetId") WHERE ((("deletedAt" IS NULL) AND ("isVisible" IS TRUE)));`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_assetId_personId_idx" ON "asset_face" ("assetId", "personId");`.execute(db);
|
||||||
|
await sql`CREATE INDEX "asset_face_personId_assetId_idx" ON "asset_face" ("personId", "assetId");`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_faceClusterId_assetId_idx";`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_faceClusterId_assetId_notDeleted_isVisible_idx";`.execute(db);
|
||||||
|
await sql`DROP INDEX "asset_face_assetId_faceClusterId_idx";`.execute(db);
|
||||||
|
await sql`ALTER TABLE "asset_face" ADD CONSTRAINT "asset_face_personId_fkey" FOREIGN KEY ("personId") REFERENCES "person" ("id") ON UPDATE CASCADE ON DELETE SET NULL;`.execute(db);
|
||||||
|
await sql`ALTER TABLE "asset_face" DROP CONSTRAINT "asset_face_faceClusterId_fkey";`.execute(db);
|
||||||
|
await sql`DROP TABLE "face_cluster";`.execute(db);
|
||||||
|
await sql`DROP TRIGGER "face_cluster_updatedAt" ON "face_cluster";`.execute(db);
|
||||||
|
await sql`INSERT INTO "migration_overrides" ("name", "value") VALUES ('index_asset_face_personId_assetId_notDeleted_isVisible_idx', '{"sql":"CREATE INDEX \\"asset_face_personId_assetId_notDeleted_isVisible_idx\\" ON \\"asset_face\\" (\\"personId\\", \\"assetId\\") WHERE (\\"deletedAt\\" IS NULL AND \\"isVisible\\" IS TRUE);","name":"asset_face_personId_assetId_notDeleted_isVisible_idx","type":"index"}'::jsonb);`.execute(db);
|
||||||
|
await sql`DELETE FROM "migration_overrides" WHERE "name" = 'trigger_face_cluster_updatedAt';`.execute(db);
|
||||||
|
await sql`DELETE FROM "migration_overrides" WHERE "name" = 'index_asset_face_faceClusterId_assetId_notDeleted_isVisible_idx';`.execute(db);
|
||||||
|
}
|
||||||
@@ -11,8 +11,8 @@ import {
|
|||||||
UpdateDateColumn,
|
UpdateDateColumn,
|
||||||
} from '@immich/sql-tools';
|
} from '@immich/sql-tools';
|
||||||
import { CreateIdColumn, UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
import { CreateIdColumn, UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
||||||
import { AlbumUserRole } from 'src/enum';
|
import { AlbumUserRole, SharingPermission } from 'src/enum';
|
||||||
import { album_user_role_enum } from 'src/schema/enums';
|
import { album_user_role_enum, sharing_permission_enum } from 'src/schema/enums';
|
||||||
import { album_user_after_insert, album_user_delete_audit } from 'src/schema/functions';
|
import { album_user_after_insert, album_user_delete_audit } from 'src/schema/functions';
|
||||||
import { AlbumTable } from 'src/schema/tables/album.table';
|
import { AlbumTable } from 'src/schema/tables/album.table';
|
||||||
import { UserTable } from 'src/schema/tables/user.table';
|
import { UserTable } from 'src/schema/tables/user.table';
|
||||||
@@ -69,4 +69,14 @@ export class AlbumUserTable {
|
|||||||
|
|
||||||
@UpdateDateColumn()
|
@UpdateDateColumn()
|
||||||
updatedAt!: Generated<Timestamp>;
|
updatedAt!: Generated<Timestamp>;
|
||||||
|
|
||||||
|
@Column({
|
||||||
|
array: true,
|
||||||
|
enum: sharing_permission_enum,
|
||||||
|
default: [SharingPermission.AssetRead, SharingPermission.ExifRead],
|
||||||
|
})
|
||||||
|
permissions!: Generated<SharingPermission[]>;
|
||||||
|
|
||||||
|
@Column({ type: 'boolean', default: false })
|
||||||
|
inTimeline!: Generated<boolean>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ import { SourceType } from 'src/enum';
|
|||||||
import { asset_face_source_type } from 'src/schema/enums';
|
import { asset_face_source_type } from 'src/schema/enums';
|
||||||
import { asset_face_audit } from 'src/schema/functions';
|
import { asset_face_audit } from 'src/schema/functions';
|
||||||
import { AssetTable } from 'src/schema/tables/asset.table';
|
import { AssetTable } from 'src/schema/tables/asset.table';
|
||||||
import { PersonTable } from 'src/schema/tables/person.table';
|
import { FaceClusterTable } from 'src/schema/tables/face-cluster.table';
|
||||||
|
|
||||||
@Table({ name: 'asset_face' })
|
@Table({ name: 'asset_face' })
|
||||||
@UpdatedAtTrigger('asset_face_updatedAt')
|
@UpdatedAtTrigger('asset_face_updatedAt')
|
||||||
@@ -26,13 +26,13 @@ import { PersonTable } from 'src/schema/tables/person.table';
|
|||||||
when: 'pg_trigger_depth() = 0',
|
when: 'pg_trigger_depth() = 0',
|
||||||
})
|
})
|
||||||
// schemaFromDatabase does not preserve column order
|
// schemaFromDatabase does not preserve column order
|
||||||
@Index({ name: 'asset_face_assetId_personId_idx', columns: ['assetId', 'personId'] })
|
@Index({ name: 'asset_face_assetId_faceClusterId_idx', columns: ['assetId', 'faceClusterId'] })
|
||||||
@Index({
|
@Index({
|
||||||
name: 'asset_face_personId_assetId_notDeleted_isVisible_idx',
|
name: 'asset_face_faceClusterId_assetId_notDeleted_isVisible_idx',
|
||||||
columns: ['personId', 'assetId'],
|
columns: ['faceClusterId', 'assetId'],
|
||||||
where: '"deletedAt" IS NULL AND "isVisible" IS TRUE',
|
where: '"deletedAt" IS NULL AND "isVisible" IS TRUE',
|
||||||
})
|
})
|
||||||
@Index({ columns: ['personId', 'assetId'] })
|
@Index({ columns: ['faceClusterId', 'assetId'] })
|
||||||
export class AssetFaceTable {
|
export class AssetFaceTable {
|
||||||
@PrimaryGeneratedColumn()
|
@PrimaryGeneratedColumn()
|
||||||
id!: Generated<string>;
|
id!: Generated<string>;
|
||||||
@@ -45,14 +45,14 @@ export class AssetFaceTable {
|
|||||||
})
|
})
|
||||||
assetId!: string;
|
assetId!: string;
|
||||||
|
|
||||||
@ForeignKeyColumn(() => PersonTable, {
|
@ForeignKeyColumn(() => FaceClusterTable, {
|
||||||
onDelete: 'SET NULL',
|
onDelete: 'SET NULL',
|
||||||
onUpdate: 'CASCADE',
|
onUpdate: 'CASCADE',
|
||||||
nullable: true,
|
nullable: true,
|
||||||
// [personId, assetId] makes this redundant
|
// [faceClusterId, assetId] makes this redundant
|
||||||
index: false,
|
index: false,
|
||||||
})
|
})
|
||||||
personId!: string | null;
|
faceClusterId!: string | null;
|
||||||
|
|
||||||
@Column({ default: 0, type: 'integer' })
|
@Column({ default: 0, type: 'integer' })
|
||||||
imageWidth!: Generated<number>;
|
imageWidth!: Generated<number>;
|
||||||
|
|||||||
@@ -0,0 +1,25 @@
|
|||||||
|
import {
|
||||||
|
CreateDateColumn,
|
||||||
|
Generated,
|
||||||
|
PrimaryGeneratedColumn,
|
||||||
|
Table,
|
||||||
|
Timestamp,
|
||||||
|
UpdateDateColumn,
|
||||||
|
} from '@immich/sql-tools';
|
||||||
|
import { UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
||||||
|
|
||||||
|
@Table('face_cluster')
|
||||||
|
@UpdatedAtTrigger('face_cluster_updatedAt')
|
||||||
|
export class FaceClusterTable {
|
||||||
|
@PrimaryGeneratedColumn('uuid')
|
||||||
|
id!: Generated<string>;
|
||||||
|
|
||||||
|
@CreateDateColumn()
|
||||||
|
createdAt!: Generated<Timestamp>;
|
||||||
|
|
||||||
|
@UpdateDateColumn()
|
||||||
|
updatedAt!: Generated<Timestamp>;
|
||||||
|
|
||||||
|
@UpdateIdColumn({ index: true })
|
||||||
|
updateId!: Generated<string>;
|
||||||
|
}
|
||||||
@@ -9,6 +9,8 @@ import {
|
|||||||
UpdateDateColumn,
|
UpdateDateColumn,
|
||||||
} from '@immich/sql-tools';
|
} from '@immich/sql-tools';
|
||||||
import { CreateIdColumn, UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
import { CreateIdColumn, UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
||||||
|
import { SharingPermission } from 'src/enum';
|
||||||
|
import { sharing_permission_enum } from 'src/schema/enums';
|
||||||
import { partner_delete_audit } from 'src/schema/functions';
|
import { partner_delete_audit } from 'src/schema/functions';
|
||||||
import { UserTable } from 'src/schema/tables/user.table';
|
import { UserTable } from 'src/schema/tables/user.table';
|
||||||
|
|
||||||
@@ -46,4 +48,7 @@ export class PartnerTable {
|
|||||||
|
|
||||||
@UpdateIdColumn({ index: true })
|
@UpdateIdColumn({ index: true })
|
||||||
updateId!: Generated<string>;
|
updateId!: Generated<string>;
|
||||||
|
|
||||||
|
@Column({ array: true, enum: sharing_permission_enum, default: [SharingPermission.All] })
|
||||||
|
permissions!: Generated<SharingPermission[]>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import {
|
|||||||
import { UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
import { UpdatedAtTrigger, UpdateIdColumn } from 'src/decorators';
|
||||||
import { person_delete_audit } from 'src/schema/functions';
|
import { person_delete_audit } from 'src/schema/functions';
|
||||||
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
import { AssetFaceTable } from 'src/schema/tables/asset-face.table';
|
||||||
|
import { FaceClusterTable } from 'src/schema/tables/face-cluster.table';
|
||||||
import { UserTable } from 'src/schema/tables/user.table';
|
import { UserTable } from 'src/schema/tables/user.table';
|
||||||
|
|
||||||
@Table('person')
|
@Table('person')
|
||||||
@@ -43,9 +44,6 @@ export class PersonTable {
|
|||||||
@ForeignKeyColumn(() => UserTable, { onDelete: 'CASCADE', onUpdate: 'CASCADE', nullable: false })
|
@ForeignKeyColumn(() => UserTable, { onDelete: 'CASCADE', onUpdate: 'CASCADE', nullable: false })
|
||||||
ownerId!: string;
|
ownerId!: string;
|
||||||
|
|
||||||
@Column({ default: '' })
|
|
||||||
name!: Generated<string>;
|
|
||||||
|
|
||||||
@Column({ default: '' })
|
@Column({ default: '' })
|
||||||
thumbnailPath!: Generated<string>;
|
thumbnailPath!: Generated<string>;
|
||||||
|
|
||||||
@@ -55,6 +53,9 @@ export class PersonTable {
|
|||||||
@Column({ type: 'date', nullable: true })
|
@Column({ type: 'date', nullable: true })
|
||||||
birthDate!: Timestamp | null;
|
birthDate!: Timestamp | null;
|
||||||
|
|
||||||
|
@Column({ default: '' })
|
||||||
|
name!: Generated<string>;
|
||||||
|
|
||||||
@ForeignKeyColumn(() => AssetFaceTable, { onDelete: 'SET NULL', nullable: true })
|
@ForeignKeyColumn(() => AssetFaceTable, { onDelete: 'SET NULL', nullable: true })
|
||||||
faceAssetId!: string | null;
|
faceAssetId!: string | null;
|
||||||
|
|
||||||
@@ -66,4 +67,7 @@ export class PersonTable {
|
|||||||
|
|
||||||
@UpdateIdColumn({ index: true })
|
@UpdateIdColumn({ index: true })
|
||||||
updateId!: Generated<string>;
|
updateId!: Generated<string>;
|
||||||
|
|
||||||
|
@ForeignKeyColumn(() => FaceClusterTable, { onDelete: 'CASCADE', onUpdate: 'CASCADE', nullable: true, index: true })
|
||||||
|
faceClusterId!: string | null;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import {
|
|||||||
CreateDateColumn,
|
CreateDateColumn,
|
||||||
DeleteDateColumn,
|
DeleteDateColumn,
|
||||||
Generated,
|
Generated,
|
||||||
|
GeneratedColumn,
|
||||||
Index,
|
Index,
|
||||||
PrimaryGeneratedColumn,
|
PrimaryGeneratedColumn,
|
||||||
Table,
|
Table,
|
||||||
@@ -82,4 +83,7 @@ export class UserTable {
|
|||||||
|
|
||||||
@UpdateIdColumn({ index: true })
|
@UpdateIdColumn({ index: true })
|
||||||
updateId!: Generated<string>;
|
updateId!: Generated<string>;
|
||||||
|
|
||||||
|
@GeneratedColumn('uuid')
|
||||||
|
trustedGroupId!: Generated<string>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,13 +8,15 @@ import {
|
|||||||
CreateAlbumDto,
|
CreateAlbumDto,
|
||||||
GetAlbumsDto,
|
GetAlbumsDto,
|
||||||
mapAlbum,
|
mapAlbum,
|
||||||
|
SharingPermissionsResponseDto,
|
||||||
UpdateAlbumDto,
|
UpdateAlbumDto,
|
||||||
UpdateAlbumUserDto,
|
UpdateAlbumUserDto,
|
||||||
|
UpdateSharingPermissionsDto,
|
||||||
} from 'src/dtos/album.dto';
|
} from 'src/dtos/album.dto';
|
||||||
import { BulkIdErrorReason, BulkIdResponseDto, BulkIdsDto } from 'src/dtos/asset-ids.response.dto';
|
import { BulkIdErrorReason, BulkIdResponseDto, BulkIdsDto } from 'src/dtos/asset-ids.response.dto';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import { MapMarkerResponseDto } from 'src/dtos/map.dto';
|
import { MapMarkerResponseDto } from 'src/dtos/map.dto';
|
||||||
import { AlbumUserRole, Permission } from 'src/enum';
|
import { AlbumUserRole, Permission, SharingPermission } from 'src/enum';
|
||||||
import { AlbumAssetCount, AlbumInfoOptions } from 'src/repositories/album.repository';
|
import { AlbumAssetCount, AlbumInfoOptions } from 'src/repositories/album.repository';
|
||||||
import { BaseService } from 'src/services/base.service';
|
import { BaseService } from 'src/services/base.service';
|
||||||
import { addAssets, removeAssets } from 'src/utils/asset.util';
|
import { addAssets, removeAssets } from 'src/utils/asset.util';
|
||||||
@@ -137,6 +139,11 @@ export class AlbumService extends BaseService {
|
|||||||
);
|
);
|
||||||
|
|
||||||
for (const { userId } of albumUsers) {
|
for (const { userId } of albumUsers) {
|
||||||
|
await this.userRepository.mergeTrustedGroups({
|
||||||
|
userId: auth.user.id,
|
||||||
|
userIdToMerge: userId,
|
||||||
|
});
|
||||||
|
|
||||||
await this.eventRepository.emit('AlbumInvite', { id: album.id, userId, senderName: auth.user.name });
|
await this.eventRepository.emit('AlbumInvite', { id: album.id, userId, senderName: auth.user.name });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -306,7 +313,17 @@ export class AlbumService extends BaseService {
|
|||||||
throw new BadRequestException('Invalid user');
|
throw new BadRequestException('Invalid user');
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.albumUserRepository.create({ userId, albumId: id, role });
|
await this.userRepository.mergeTrustedGroups({
|
||||||
|
userId: auth.user.id,
|
||||||
|
userIdToMerge: userId,
|
||||||
|
});
|
||||||
|
await this.albumUserRepository.create({
|
||||||
|
userId,
|
||||||
|
albumId: id,
|
||||||
|
role,
|
||||||
|
permissions: [SharingPermission.AssetRead, SharingPermission.ExifRead],
|
||||||
|
});
|
||||||
|
|
||||||
await this.eventRepository.emit('AlbumInvite', { id, userId, senderName: auth.user.name });
|
await this.eventRepository.emit('AlbumInvite', { id, userId, senderName: auth.user.name });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -345,6 +362,19 @@ export class AlbumService extends BaseService {
|
|||||||
await this.albumUserRepository.update({ albumId: id, userId }, { role: dto.role });
|
await this.albumUserRepository.update({ albumId: id, userId }, { role: dto.role });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async updateSelf(auth: AuthDto, albumId: string, dto: UpdateSharingPermissionsDto): Promise<void> {
|
||||||
|
await this.requireAccess({ auth, permission: Permission.AlbumAssetCreate, ids: [albumId] });
|
||||||
|
await this.albumUserRepository.update(
|
||||||
|
{ albumId, userId: auth.user.id },
|
||||||
|
{ permissions: dto.permissions, inTimeline: dto.inTimeline },
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getSelf(auth: AuthDto, albumId: string): Promise<SharingPermissionsResponseDto> {
|
||||||
|
await this.requireAccess({ auth, permission: Permission.AlbumAssetCreate, ids: [albumId] });
|
||||||
|
return this.albumUserRepository.get({ userId: auth.user.id, albumId });
|
||||||
|
}
|
||||||
|
|
||||||
private async findOrFail(id: string, authUserId: string, options: AlbumInfoOptions) {
|
private async findOrFail(id: string, authUserId: string, options: AlbumInfoOptions) {
|
||||||
const album = await this.albumRepository.getById(id, options, authUserId);
|
const album = await this.albumRepository.getById(id, options, authUserId);
|
||||||
if (!album) {
|
if (!album) {
|
||||||
|
|||||||
@@ -32,10 +32,11 @@ import {
|
|||||||
JobStatus,
|
JobStatus,
|
||||||
Permission,
|
Permission,
|
||||||
QueueName,
|
QueueName,
|
||||||
|
SharingPermission,
|
||||||
} from 'src/enum';
|
} from 'src/enum';
|
||||||
import { BaseService } from 'src/services/base.service';
|
import { BaseService } from 'src/services/base.service';
|
||||||
import { JobItem, JobOf } from 'src/types';
|
import { JobItem, JobOf } from 'src/types';
|
||||||
import { requireElevatedPermission } from 'src/utils/access';
|
import { hasPermissions, requireElevatedPermission } from 'src/utils/access';
|
||||||
import {
|
import {
|
||||||
getAssetFiles,
|
getAssetFiles,
|
||||||
getDimensions,
|
getDimensions,
|
||||||
@@ -62,14 +63,18 @@ export class AssetService extends BaseService {
|
|||||||
async get(auth: AuthDto, id: string): Promise<AssetResponseDto | SanitizedAssetResponseDto> {
|
async get(auth: AuthDto, id: string): Promise<AssetResponseDto | SanitizedAssetResponseDto> {
|
||||||
await this.requireAccess({ auth, permission: Permission.AssetRead, ids: [id] });
|
await this.requireAccess({ auth, permission: Permission.AssetRead, ids: [id] });
|
||||||
|
|
||||||
const asset = await this.assetRepository.getById(id, {
|
const asset = await this.assetRepository.getById(
|
||||||
exifInfo: true,
|
id,
|
||||||
owner: true,
|
{
|
||||||
faces: { person: true },
|
exifInfo: true,
|
||||||
stack: { assets: true },
|
owner: true,
|
||||||
edits: true,
|
faces: { person: true },
|
||||||
tags: true,
|
stack: { assets: true },
|
||||||
});
|
edits: true,
|
||||||
|
tags: true,
|
||||||
|
},
|
||||||
|
auth.user.id,
|
||||||
|
);
|
||||||
|
|
||||||
if (!asset) {
|
if (!asset) {
|
||||||
throw new BadRequestException('Asset not found');
|
throw new BadRequestException('Asset not found');
|
||||||
@@ -85,7 +90,7 @@ export class AssetService extends BaseService {
|
|||||||
delete data.owner;
|
delete data.owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.ownerId !== auth.user.id || auth.sharedLink) {
|
if (!hasPermissions(data, SharingPermission.PersonRead)) {
|
||||||
data.people = [];
|
data.people = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -85,7 +85,11 @@ export class NotificationService extends BaseService {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.logger.error(`Unable to run job handler (${job.name}): ${error}`, error?.stack, JSON.stringify(job.data));
|
this.logger.error(
|
||||||
|
`Unable to run job handler (${job.name}): ${error}`,
|
||||||
|
error?.stack,
|
||||||
|
'data' in job ? JSON.stringify(job.data) : {},
|
||||||
|
);
|
||||||
|
|
||||||
switch (job.name) {
|
switch (job.name) {
|
||||||
case JobName.DatabaseBackup: {
|
case JobName.DatabaseBackup: {
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { Partner } from 'src/database';
|
|||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import { PartnerCreateDto, PartnerResponseDto, PartnerSearchDto, PartnerUpdateDto } from 'src/dtos/partner.dto';
|
import { PartnerCreateDto, PartnerResponseDto, PartnerSearchDto, PartnerUpdateDto } from 'src/dtos/partner.dto';
|
||||||
import { mapUser } from 'src/dtos/user.dto';
|
import { mapUser } from 'src/dtos/user.dto';
|
||||||
import { Permission } from 'src/enum';
|
import { JobName, Permission, SharingPermission } from 'src/enum';
|
||||||
import { PartnerDirection, PartnerIds } from 'src/repositories/partner.repository';
|
import { PartnerDirection, PartnerIds } from 'src/repositories/partner.repository';
|
||||||
import { BaseService } from 'src/services/base.service';
|
import { BaseService } from 'src/services/base.service';
|
||||||
|
|
||||||
@@ -16,7 +16,15 @@ export class PartnerService extends BaseService {
|
|||||||
throw new BadRequestException(`Partner already exists`);
|
throw new BadRequestException(`Partner already exists`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const partner = await this.partnerRepository.create(partnerId);
|
const { numUpdatedRows } = await this.userRepository.mergeTrustedGroups({
|
||||||
|
userId: auth.user.id,
|
||||||
|
userIdToMerge: sharedWithId,
|
||||||
|
});
|
||||||
|
const partner = await this.partnerRepository.create({ ...partnerId, permissions: [SharingPermission.All] });
|
||||||
|
if (numUpdatedRows > 0) {
|
||||||
|
await this.jobRepository.queue({ name: JobName.FacialRecognitionMerge, data: { id: sharedWithId } });
|
||||||
|
}
|
||||||
|
|
||||||
return this.mapPartner(partner, PartnerDirection.SharedBy);
|
return this.mapPartner(partner, PartnerDirection.SharedBy);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -28,6 +36,10 @@ export class PartnerService extends BaseService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
await this.partnerRepository.remove(partnerId);
|
await this.partnerRepository.remove(partnerId);
|
||||||
|
const { numUpdatedRows } = await this.userRepository.updateTrustedGroups(auth.user.id);
|
||||||
|
if (numUpdatedRows > 0) {
|
||||||
|
await this.jobRepository.queue({ name: JobName.FacialRecognitionQueueAll, data: { force: true } });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async search(auth: AuthDto, { direction }: PartnerSearchDto): Promise<PartnerResponseDto[]> {
|
async search(auth: AuthDto, { direction }: PartnerSearchDto): Promise<PartnerResponseDto[]> {
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
FaceDto,
|
FaceDto,
|
||||||
mapFaces,
|
mapFaces,
|
||||||
mapPerson,
|
mapPerson,
|
||||||
MergePersonDto,
|
MergeFaceClusterDto,
|
||||||
PeopleResponseDto,
|
PeopleResponseDto,
|
||||||
PeopleUpdateDto,
|
PeopleUpdateDto,
|
||||||
PersonCreateDto,
|
PersonCreateDto,
|
||||||
@@ -127,11 +127,11 @@ export class PersonService extends BaseService {
|
|||||||
|
|
||||||
async getFacesById(auth: AuthDto, dto: FaceDto): Promise<AssetFaceResponseDto[]> {
|
async getFacesById(auth: AuthDto, dto: FaceDto): Promise<AssetFaceResponseDto[]> {
|
||||||
await this.requireAccess({ auth, permission: Permission.AssetRead, ids: [dto.id] });
|
await this.requireAccess({ auth, permission: Permission.AssetRead, ids: [dto.id] });
|
||||||
const faces = await this.personRepository.getFaces(dto.id);
|
const faces = await this.personRepository.getFaces(dto.id, { userId: auth.user.id });
|
||||||
const asset = await this.assetRepository.getForFaces(dto.id);
|
const asset = await this.assetRepository.getForFaces(dto.id);
|
||||||
const assetDimensions = getDimensions(asset);
|
const assetDimensions = getDimensions(asset);
|
||||||
|
|
||||||
return faces.map((face) => mapFaces(face, auth, asset.edits, assetDimensions));
|
return faces.map((face) => mapFaces(face, asset.edits, assetDimensions));
|
||||||
}
|
}
|
||||||
|
|
||||||
async createNewFeaturePhoto(changeFeaturePhoto: string[]) {
|
async createNewFeaturePhoto(changeFeaturePhoto: string[]) {
|
||||||
@@ -159,7 +159,7 @@ export class PersonService extends BaseService {
|
|||||||
|
|
||||||
async getStatistics(auth: AuthDto, id: string): Promise<PersonStatisticsResponseDto> {
|
async getStatistics(auth: AuthDto, id: string): Promise<PersonStatisticsResponseDto> {
|
||||||
await this.requireAccess({ auth, permission: Permission.PersonRead, ids: [id] });
|
await this.requireAccess({ auth, permission: Permission.PersonRead, ids: [id] });
|
||||||
return this.personRepository.getStatistics(id);
|
return this.personRepository.getStatistics(auth.user.id, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getThumbnail(auth: AuthDto, id: string): Promise<ImmichFileResponse> {
|
async getThumbnail(auth: AuthDto, id: string): Promise<ImmichFileResponse> {
|
||||||
@@ -438,7 +438,7 @@ export class PersonService extends BaseService {
|
|||||||
|
|
||||||
const lastRun = new Date().toISOString();
|
const lastRun = new Date().toISOString();
|
||||||
const facePagination = this.personRepository.getAllFaces(
|
const facePagination = this.personRepository.getAllFaces(
|
||||||
force ? undefined : { personId: null, sourceType: SourceType.MachineLearning },
|
force ? undefined : { faceClusterId: null, sourceType: SourceType.MachineLearning },
|
||||||
);
|
);
|
||||||
|
|
||||||
let jobs: { name: JobName.FacialRecognition; data: { id: string; deferred: false } }[] = [];
|
let jobs: { name: JobName.FacialRecognition; data: { id: string; deferred: false } }[] = [];
|
||||||
@@ -481,8 +481,8 @@ export class PersonService extends BaseService {
|
|||||||
return JobStatus.Failed;
|
return JobStatus.Failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (face.personId) {
|
if (face.faceClusterId) {
|
||||||
this.logger.debug(`Face ${id} already has a person assigned`);
|
this.logger.debug(`Face ${id} already belongs to a face cluster`);
|
||||||
return JobStatus.Skipped;
|
return JobStatus.Skipped;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -511,8 +511,8 @@ export class PersonService extends BaseService {
|
|||||||
return JobStatus.Skipped;
|
return JobStatus.Skipped;
|
||||||
}
|
}
|
||||||
|
|
||||||
let personId = matches.find((match) => match.personId)?.personId;
|
let faceClusterId = matches.find((match) => match.faceClusterId)?.faceClusterId;
|
||||||
if (!personId) {
|
if (!faceClusterId) {
|
||||||
const matchWithPerson = await this.searchRepository.searchFaces({
|
const matchWithPerson = await this.searchRepository.searchFaces({
|
||||||
userIds: [face.asset.ownerId],
|
userIds: [face.asset.ownerId],
|
||||||
embedding: face.faceSearch.embedding,
|
embedding: face.faceSearch.embedding,
|
||||||
@@ -523,20 +523,108 @@ export class PersonService extends BaseService {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (matchWithPerson.length > 0) {
|
if (matchWithPerson.length > 0) {
|
||||||
personId = matchWithPerson[0].personId;
|
faceClusterId = matchWithPerson[0].faceClusterId;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isCore && !personId) {
|
if (isCore && !faceClusterId) {
|
||||||
this.logger.log(`Creating new person for face ${id}`);
|
this.logger.log(`Creating new person for face ${id}`);
|
||||||
const newPerson = await this.personRepository.create({ ownerId: face.asset.ownerId, faceAssetId: face.id });
|
const newPerson = await this.personRepository.create({ ownerId: face.asset.ownerId, faceAssetId: face.id });
|
||||||
await this.jobRepository.queue({ name: JobName.PersonGenerateThumbnail, data: { id: newPerson.id } });
|
await this.jobRepository.queue({ name: JobName.PersonGenerateThumbnail, data: { id: newPerson.id } });
|
||||||
personId = newPerson.id;
|
faceClusterId = newPerson.faceClusterId;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (personId) {
|
if (faceClusterId) {
|
||||||
this.logger.debug(`Assigning face ${id} to person ${personId}`);
|
this.logger.debug(`Assigning face ${id} to face cluster ${faceClusterId}`);
|
||||||
await this.personRepository.reassignFaces({ faceIds: [id], newPersonId: personId });
|
await this.personRepository.reassignFaces({ faceIds: [id], newFaceClusterId: faceClusterId });
|
||||||
|
}
|
||||||
|
|
||||||
|
return JobStatus.Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
@OnJob({ name: JobName.FacialRecognitionMerge, queue: QueueName.FacialRecognition })
|
||||||
|
async mergeClusters({ id: userId }: JobOf<JobName.FacialRecognitionMerge>): Promise<JobStatus> {
|
||||||
|
const { machineLearning } = await this.getConfig({ withCache: true });
|
||||||
|
if (!isFacialRecognitionEnabled(machineLearning)) {
|
||||||
|
return JobStatus.Skipped;
|
||||||
|
}
|
||||||
|
|
||||||
|
const faces = this.personRepository.getAllFaces({ sourceType: SourceType.MachineLearning });
|
||||||
|
for await (const { id } of faces) {
|
||||||
|
const face = await this.personRepository.getFaceForFacialRecognitionJob(id);
|
||||||
|
if (!face?.faceSearch || !face.asset) {
|
||||||
|
this.logger.warn(`Face ${id} does not have an embedding`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (face.asset.ownerId === userId) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let faceClusterId: string | null = null;
|
||||||
|
const matchWithPerson = await this.searchRepository.searchFaces({
|
||||||
|
userIds: [face.asset.ownerId],
|
||||||
|
embedding: face.faceSearch.embedding,
|
||||||
|
maxDistance: machineLearning.facialRecognition.maxDistance,
|
||||||
|
numResults: 100,
|
||||||
|
hasPerson: true,
|
||||||
|
minBirthDate: new Date(face.asset.fileCreatedAt),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (matchWithPerson.length > 0) {
|
||||||
|
// favor a person that's not owned by us to merge people with a newly shared with user
|
||||||
|
// probably do smarter stuff here like pick the person with a name, if both have a name set aliases or whatever
|
||||||
|
const match = matchWithPerson.find((match) => match.ownerId !== userId) ?? matchWithPerson[0];
|
||||||
|
if (match.faceClusterId && face.asset.ownerId !== match.ownerId) {
|
||||||
|
// TODO should probably be a DB constraint?
|
||||||
|
const people = await this.personRepository.getByFaceClusterId(match.faceClusterId);
|
||||||
|
if (!people.some((person) => person.ownerId === face.asset?.ownerId)) {
|
||||||
|
const person = await this.personRepository.create({
|
||||||
|
ownerId: face.asset.ownerId,
|
||||||
|
faceClusterId: match.faceClusterId,
|
||||||
|
});
|
||||||
|
await this.createNewFeaturePhoto([person.id]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
faceClusterId = match.faceClusterId;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!faceClusterId) {
|
||||||
|
const matches = await this.searchRepository.searchFaces({
|
||||||
|
userIds: [userId],
|
||||||
|
embedding: face.faceSearch.embedding,
|
||||||
|
maxDistance: machineLearning.facialRecognition.maxDistance,
|
||||||
|
numResults: machineLearning.facialRecognition.minFaces,
|
||||||
|
minBirthDate: new Date(face.asset.fileCreatedAt),
|
||||||
|
});
|
||||||
|
|
||||||
|
const match = matches.find((match) => match.faceClusterId);
|
||||||
|
if (
|
||||||
|
match &&
|
||||||
|
match.faceClusterId &&
|
||||||
|
face.asset.ownerId !== match.ownerId &&
|
||||||
|
matches.length >= machineLearning.facialRecognition.minFaces
|
||||||
|
) {
|
||||||
|
// TODO should probably be a DB constraint?
|
||||||
|
const people = await this.personRepository.getByFaceClusterId(match.faceClusterId);
|
||||||
|
|
||||||
|
if (!people.some((person) => person.ownerId === face.asset?.ownerId)) {
|
||||||
|
const person = await this.personRepository.create({
|
||||||
|
ownerId: face.asset.ownerId,
|
||||||
|
faceClusterId: match.faceClusterId,
|
||||||
|
});
|
||||||
|
await this.createNewFeaturePhoto([person.id]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
faceClusterId = match?.faceClusterId ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (faceClusterId) {
|
||||||
|
this.logger.log(`Assigning face ${id} to face cluster ${faceClusterId}`);
|
||||||
|
await this.personRepository.reassignFaces({ faceIds: [id], newFaceClusterId: faceClusterId });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return JobStatus.Success;
|
return JobStatus.Success;
|
||||||
@@ -554,7 +642,7 @@ export class PersonService extends BaseService {
|
|||||||
return JobStatus.Success;
|
return JobStatus.Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
async mergePerson(auth: AuthDto, id: string, dto: MergePersonDto): Promise<BulkIdResponseDto[]> {
|
async mergePerson(auth: AuthDto, id: string, dto: MergeFaceClusterDto): Promise<BulkIdResponseDto[]> {
|
||||||
const mergeIds = dto.ids;
|
const mergeIds = dto.ids;
|
||||||
if (mergeIds.includes(id)) {
|
if (mergeIds.includes(id)) {
|
||||||
throw new BadRequestException('Cannot merge a person into themselves');
|
throw new BadRequestException('Cannot merge a person into themselves');
|
||||||
@@ -600,7 +688,7 @@ export class PersonService extends BaseService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const mergeName = mergePerson.name || mergePerson.id;
|
const mergeName = mergePerson.name || mergePerson.id;
|
||||||
const mergeData: UpdateFacesData = { oldPersonId: mergeId, newPersonId: id };
|
const mergeData: UpdateFacesData = { oldFaceClusterId: mergeId, newFaceClusterId: id };
|
||||||
this.logger.log(`Merging ${mergeName} into ${primaryName}`);
|
this.logger.log(`Merging ${mergeName} into ${primaryName}`);
|
||||||
|
|
||||||
await this.personRepository.reassignFaces(mergeData);
|
await this.personRepository.reassignFaces(mergeData);
|
||||||
@@ -613,6 +701,7 @@ export class PersonService extends BaseService {
|
|||||||
results.push({ id: mergeId, success: false, error: BulkIdErrorReason.UNKNOWN });
|
results.push({ id: mergeId, success: false, error: BulkIdErrorReason.UNKNOWN });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -682,8 +771,12 @@ export class PersonService extends BaseService {
|
|||||||
dto.imageHeight = originalDimensions.height;
|
dto.imageHeight = originalDimensions.height;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!person?.faceClusterId) {
|
||||||
|
throw new Error('Person must already have some recognized faces and belong to a face cluster');
|
||||||
|
}
|
||||||
|
|
||||||
await this.personRepository.createAssetFace({
|
await this.personRepository.createAssetFace({
|
||||||
personId: dto.personId,
|
faceClusterId: person.faceClusterId,
|
||||||
assetId: dto.assetId,
|
assetId: dto.assetId,
|
||||||
imageHeight: dto.imageHeight,
|
imageHeight: dto.imageHeight,
|
||||||
imageWidth: dto.imageWidth,
|
imageWidth: dto.imageWidth,
|
||||||
|
|||||||
@@ -208,6 +208,7 @@ export class SearchService extends BaseService {
|
|||||||
repository: this.partnerRepository,
|
repository: this.partnerRepository,
|
||||||
timelineEnabled: true,
|
timelineEnabled: true,
|
||||||
});
|
});
|
||||||
|
console.log(auth.user.id, partnerIds);
|
||||||
return [auth.user.id, ...partnerIds];
|
return [auth.user.id, ...partnerIds];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+4
-1
@@ -204,7 +204,9 @@ export type ConcurrentQueueName = Exclude<
|
|||||||
| QueueName.BackupDatabase
|
| QueueName.BackupDatabase
|
||||||
>;
|
>;
|
||||||
|
|
||||||
export type Jobs = { [K in JobItem['name']]: (JobItem & { name: K })['data'] };
|
export type Jobs = {
|
||||||
|
[K in JobItem['name']]: 'data' extends keyof (JobItem & { name: K }) ? (JobItem & { name: K })['data'] : never;
|
||||||
|
};
|
||||||
export type JobOf<T extends JobName> = Jobs[T];
|
export type JobOf<T extends JobName> = Jobs[T];
|
||||||
|
|
||||||
export interface IBaseJob {
|
export interface IBaseJob {
|
||||||
@@ -351,6 +353,7 @@ export type JobItem =
|
|||||||
| { name: JobName.AssetDetectFaces; data: IEntityJob }
|
| { name: JobName.AssetDetectFaces; data: IEntityJob }
|
||||||
| { name: JobName.FacialRecognitionQueueAll; data: INightlyJob }
|
| { name: JobName.FacialRecognitionQueueAll; data: INightlyJob }
|
||||||
| { name: JobName.FacialRecognition; data: IDeferrableJob }
|
| { name: JobName.FacialRecognition; data: IDeferrableJob }
|
||||||
|
| { name: JobName.FacialRecognitionMerge; data: IEntityJob }
|
||||||
| { name: JobName.PersonGenerateThumbnail; data: IEntityJob }
|
| { name: JobName.PersonGenerateThumbnail; data: IEntityJob }
|
||||||
|
|
||||||
// Smart Search
|
// Smart Search
|
||||||
|
|||||||
+82
-22
@@ -1,7 +1,7 @@
|
|||||||
import { BadRequestException, UnauthorizedException } from '@nestjs/common';
|
import { BadRequestException, UnauthorizedException } from '@nestjs/common';
|
||||||
import { AuthSharedLink } from 'src/database';
|
import { AuthSharedLink } from 'src/database';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import { AlbumUserRole, Permission } from 'src/enum';
|
import { AlbumUserRole, Permission, SharingPermission } from 'src/enum';
|
||||||
import { AccessRepository } from 'src/repositories/access.repository';
|
import { AccessRepository } from 'src/repositories/access.repository';
|
||||||
import { setDifference, setIsEqual, setIsSuperset, setUnion } from 'src/utils/set';
|
import { setDifference, setIsEqual, setIsSuperset, setUnion } from 'src/utils/set';
|
||||||
|
|
||||||
@@ -115,37 +115,41 @@ const checkOtherAccess = async (access: AccessRepository, request: OtherAccessRe
|
|||||||
|
|
||||||
case Permission.AssetRead: {
|
case Permission.AssetRead: {
|
||||||
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
const isAlbum = await access.asset.checkAlbumAccess(auth.user.id, setDifference(ids, isOwner));
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetRead]);
|
||||||
const isPartner = await access.asset.checkPartnerAccess(auth.user.id, setDifference(ids, isOwner, isAlbum));
|
return setUnion(isOwner, isShared);
|
||||||
return setUnion(isOwner, isAlbum, isPartner);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetShare: {
|
case Permission.AssetShare: {
|
||||||
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, false);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, false);
|
||||||
const isPartner = await access.asset.checkPartnerAccess(auth.user.id, setDifference(ids, isOwner));
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetShare]);
|
||||||
return setUnion(isOwner, isPartner);
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetView: {
|
case Permission.AssetView: {
|
||||||
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
const isAlbum = await access.asset.checkAlbumAccess(auth.user.id, setDifference(ids, isOwner));
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetRead]);
|
||||||
const isPartner = await access.asset.checkPartnerAccess(auth.user.id, setDifference(ids, isOwner, isAlbum));
|
return setUnion(isOwner, isShared);
|
||||||
return setUnion(isOwner, isAlbum, isPartner);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetDownload: {
|
case Permission.AssetDownload: {
|
||||||
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
const isAlbum = await access.asset.checkAlbumAccess(auth.user.id, setDifference(ids, isOwner));
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [
|
||||||
const isPartner = await access.asset.checkPartnerAccess(auth.user.id, setDifference(ids, isOwner, isAlbum));
|
SharingPermission.AssetRead,
|
||||||
return setUnion(isOwner, isAlbum, isPartner);
|
SharingPermission.ExifRead,
|
||||||
|
]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetUpdate: {
|
case Permission.AssetUpdate: {
|
||||||
return await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetUpdate]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetDelete: {
|
case Permission.AssetDelete: {
|
||||||
return await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetDelete]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetCopy: {
|
case Permission.AssetCopy: {
|
||||||
@@ -153,15 +157,21 @@ const checkOtherAccess = async (access: AccessRepository, request: OtherAccessRe
|
|||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetEditGet: {
|
case Permission.AssetEditGet: {
|
||||||
return await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetEdit]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetEditCreate: {
|
case Permission.AssetEditCreate: {
|
||||||
return await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetEdit]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AssetEditDelete: {
|
case Permission.AssetEditDelete: {
|
||||||
return await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
const isOwner = await access.asset.checkOwnerAccess(auth.user.id, ids, auth.session?.hasElevatedPermission);
|
||||||
|
const isShared = await access.asset.checkSharedAccess(auth.user.id, ids, [SharingPermission.AssetEdit]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.AlbumRead: {
|
case Permission.AlbumRead: {
|
||||||
@@ -246,7 +256,11 @@ const checkOtherAccess = async (access: AccessRepository, request: OtherAccessRe
|
|||||||
}
|
}
|
||||||
|
|
||||||
case Permission.FaceDelete: {
|
case Permission.FaceDelete: {
|
||||||
return access.person.checkFaceOwnerAccess(auth.user.id, ids);
|
const isOwner = await access.person.checkFaceOwnerAccess(auth.user.id, ids);
|
||||||
|
const isShared = await access.person.checkSharedFaceAccess(auth.user.id, setDifference(ids, isOwner), [
|
||||||
|
SharingPermission.AssetUpdate,
|
||||||
|
]);
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.NotificationRead:
|
case Permission.NotificationRead:
|
||||||
@@ -288,11 +302,40 @@ const checkOtherAccess = async (access: AccessRepository, request: OtherAccessRe
|
|||||||
return access.person.checkFaceOwnerAccess(auth.user.id, ids);
|
return access.person.checkFaceOwnerAccess(auth.user.id, ids);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.PersonRead:
|
case Permission.PersonRead: {
|
||||||
case Permission.PersonUpdate:
|
const isOwner = await access.person.checkOwnerAccess(auth.user.id, ids);
|
||||||
case Permission.PersonDelete:
|
const isShared = await access.person.checkSharedAccess(auth.user.id, setDifference(ids, isOwner), [
|
||||||
|
SharingPermission.PersonRead,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
|
}
|
||||||
|
|
||||||
case Permission.PersonMerge: {
|
case Permission.PersonMerge: {
|
||||||
return await access.person.checkOwnerAccess(auth.user.id, ids);
|
const isOwner = await access.person.checkOwnerAccess(auth.user.id, ids);
|
||||||
|
const isShared = await access.person.checkSharedAccess(auth.user.id, setDifference(ids, isOwner), [
|
||||||
|
SharingPermission.PersonMerge,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
|
}
|
||||||
|
|
||||||
|
case Permission.PersonUpdate: {
|
||||||
|
const isOwner = await access.person.checkOwnerAccess(auth.user.id, ids);
|
||||||
|
const isShared = await access.person.checkSharedAccess(auth.user.id, setDifference(ids, isOwner), [
|
||||||
|
SharingPermission.PersonUpdate,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
|
}
|
||||||
|
|
||||||
|
case Permission.PersonDelete: {
|
||||||
|
const isOwner = await access.person.checkOwnerAccess(auth.user.id, ids);
|
||||||
|
const isShared = await access.person.checkSharedAccess(auth.user.id, setDifference(ids, isOwner), [
|
||||||
|
SharingPermission.PersonDelete,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return setUnion(isOwner, isShared);
|
||||||
}
|
}
|
||||||
|
|
||||||
case Permission.PersonReassign: {
|
case Permission.PersonReassign: {
|
||||||
@@ -339,3 +382,20 @@ export const requireElevatedPermission = (auth: AuthDto) => {
|
|||||||
throw new UnauthorizedException('Elevated permission is required');
|
throw new UnauthorizedException('Elevated permission is required');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const hasPermissions = (
|
||||||
|
assetLike: { permissions: SharingPermission[] },
|
||||||
|
...permissions: SharingPermission[]
|
||||||
|
) => {
|
||||||
|
if (assetLike.permissions.includes(SharingPermission.All)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const permission of permissions) {
|
||||||
|
if (!assetLike.permissions.includes(permission)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { AssetFile } from 'src/database';
|
|||||||
import { BulkIdErrorReason, BulkIdResponseDto } from 'src/dtos/asset-ids.response.dto';
|
import { BulkIdErrorReason, BulkIdResponseDto } from 'src/dtos/asset-ids.response.dto';
|
||||||
import { UploadFieldName } from 'src/dtos/asset-media.dto';
|
import { UploadFieldName } from 'src/dtos/asset-media.dto';
|
||||||
import { AuthDto } from 'src/dtos/auth.dto';
|
import { AuthDto } from 'src/dtos/auth.dto';
|
||||||
import { AssetFileType, AssetType, AssetVisibility, Permission } from 'src/enum';
|
import { AssetFileType, AssetType, AssetVisibility, Permission, SharingPermission } from 'src/enum';
|
||||||
import { AuthRequest } from 'src/middleware/auth.guard';
|
import { AuthRequest } from 'src/middleware/auth.guard';
|
||||||
import { AccessRepository } from 'src/repositories/access.repository';
|
import { AccessRepository } from 'src/repositories/access.repository';
|
||||||
import { AssetRepository } from 'src/repositories/asset.repository';
|
import { AssetRepository } from 'src/repositories/asset.repository';
|
||||||
@@ -134,6 +134,11 @@ export const getMyPartnerIds = async ({ userId, repository, timelineEnabled }: P
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const permissions = [SharingPermission.All, SharingPermission.AssetRead];
|
||||||
|
if (!permissions.some((permission) => partner.permissions.includes(permission))) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
partnerIds.add(partner.sharedById);
|
partnerIds.add(partner.sharedById);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,9 +15,17 @@ import {
|
|||||||
import { PostgresJSDialect } from 'kysely-postgres-js';
|
import { PostgresJSDialect } from 'kysely-postgres-js';
|
||||||
import { jsonArrayFrom, jsonObjectFrom } from 'kysely/helpers/postgres';
|
import { jsonArrayFrom, jsonObjectFrom } from 'kysely/helpers/postgres';
|
||||||
import { Notice, PostgresError } from 'postgres';
|
import { Notice, PostgresError } from 'postgres';
|
||||||
import { columns, lockableProperties, LockableProperty, Person } from 'src/database';
|
import { columns, lockableProperties, LockableProperty } from 'src/database';
|
||||||
import { AssetEditActionItem } from 'src/dtos/editing.dto';
|
import { AssetEditActionItem } from 'src/dtos/editing.dto';
|
||||||
import { AssetFileType, AssetOrderBy, AssetVisibility, DatabaseExtension, ExifOrientation } from 'src/enum';
|
import {
|
||||||
|
AssetFileType,
|
||||||
|
AssetOrderBy,
|
||||||
|
AssetVisibility,
|
||||||
|
DatabaseExtension,
|
||||||
|
ExifOrientation,
|
||||||
|
SharingPermission,
|
||||||
|
} from 'src/enum';
|
||||||
|
import { hasAssetPermissions } from 'src/repositories/asset.repository';
|
||||||
import { AssetSearchBuilderOptions } from 'src/repositories/search.repository';
|
import { AssetSearchBuilderOptions } from 'src/repositories/search.repository';
|
||||||
import { DB } from 'src/schema';
|
import { DB } from 'src/schema';
|
||||||
import { AssetExifTable } from 'src/schema/tables/asset-exif.table';
|
import { AssetExifTable } from 'src/schema/tables/asset-exif.table';
|
||||||
@@ -212,19 +220,22 @@ export function withFilePath(eb: ExpressionBuilder<DB, 'asset'>, type: AssetFile
|
|||||||
|
|
||||||
export function withFacesAndPeople(
|
export function withFacesAndPeople(
|
||||||
eb: ExpressionBuilder<DB, 'asset'>,
|
eb: ExpressionBuilder<DB, 'asset'>,
|
||||||
withHidden?: boolean,
|
{ withHidden, withDeletedFace, userId: _ }: { withHidden?: boolean; withDeletedFace?: boolean; userId?: string } = {},
|
||||||
withDeletedFace?: boolean,
|
|
||||||
) {
|
) {
|
||||||
return jsonArrayFrom(
|
return jsonArrayFrom(
|
||||||
eb
|
eb
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.leftJoinLateral(
|
.select((eb) =>
|
||||||
(eb) =>
|
jsonObjectFrom(
|
||||||
eb.selectFrom('person').selectAll('person').whereRef('asset_face.personId', '=', 'person.id').as('person'),
|
eb
|
||||||
(join) => join.onTrue(),
|
.selectFrom('face_cluster')
|
||||||
|
.whereRef('face_cluster.id', '=', 'asset_face.faceClusterId')
|
||||||
|
.innerJoin('person', 'person.faceClusterId', 'face_cluster.id')
|
||||||
|
.selectAll('person')
|
||||||
|
.limit(1),
|
||||||
|
).as('person'),
|
||||||
)
|
)
|
||||||
.selectAll('asset_face')
|
.selectAll('asset_face')
|
||||||
.select((eb) => eb.table('person').$castTo<ShallowDehydrateObject<Person>>().as('person'))
|
|
||||||
.whereRef('asset_face.assetId', '=', 'asset.id')
|
.whereRef('asset_face.assetId', '=', 'asset.id')
|
||||||
.$if(!withDeletedFace, (qb) => qb.where('asset_face.deletedAt', 'is', null))
|
.$if(!withDeletedFace, (qb) => qb.where('asset_face.deletedAt', 'is', null))
|
||||||
.$if(!withHidden, (qb) => qb.where('asset_face.isVisible', 'is', true)),
|
.$if(!withHidden, (qb) => qb.where('asset_face.isVisible', 'is', true)),
|
||||||
@@ -237,11 +248,12 @@ export function hasPeople<O>(qb: SelectQueryBuilder<DB, 'asset', O>, personIds:
|
|||||||
eb
|
eb
|
||||||
.selectFrom('asset_face')
|
.selectFrom('asset_face')
|
||||||
.select('assetId')
|
.select('assetId')
|
||||||
.where('personId', '=', anyUuid(personIds!))
|
.innerJoin('person', 'person.faceClusterId', 'asset_face.faceClusterId')
|
||||||
|
.where('person.id', '=', anyUuid(personIds!))
|
||||||
.where('deletedAt', 'is', null)
|
.where('deletedAt', 'is', null)
|
||||||
.where('isVisible', 'is', true)
|
.where('isVisible', 'is', true)
|
||||||
.groupBy('assetId')
|
.groupBy('assetId')
|
||||||
.having((eb) => eb.fn.count('personId').distinct(), '=', personIds.length)
|
.having((eb) => eb.fn.count('person.id').distinct(), '=', personIds.length)
|
||||||
.as('has_people'),
|
.as('has_people'),
|
||||||
(join) => join.onRef('has_people.assetId', '=', 'asset.id'),
|
(join) => join.onRef('has_people.assetId', '=', 'asset.id'),
|
||||||
);
|
);
|
||||||
@@ -302,6 +314,30 @@ export function truncatedDate<O>(order: AssetOrderBy = AssetOrderBy.TakenAt) {
|
|||||||
return sql<O>`date_trunc(${sql.lit('MONTH')}, ${sql.ref(order === AssetOrderBy.CreatedAt ? 'asset.createdAt' : 'localDateTime')} AT TIME ZONE 'UTC') AT TIME ZONE 'UTC'`;
|
return sql<O>`date_trunc(${sql.lit('MONTH')}, ${sql.ref(order === AssetOrderBy.CreatedAt ? 'asset.createdAt' : 'localDateTime')} AT TIME ZONE 'UTC') AT TIME ZONE 'UTC'`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function withPermissions(userId: string) {
|
||||||
|
return (eb: ExpressionBuilder<DB, 'asset'>) =>
|
||||||
|
jsonArrayFrom(
|
||||||
|
eb
|
||||||
|
.selectFrom('album_user')
|
||||||
|
.select((eb) => eb.fn<SharingPermission>('unnest', ['album_user.permissions']).as('permission'))
|
||||||
|
.distinct()
|
||||||
|
.innerJoin('album_asset', 'album_user.albumId', 'album_asset.albumId')
|
||||||
|
.whereRef('album_asset.assetId', '=', 'asset.id')
|
||||||
|
.whereRef('album_user.userId', '=', 'asset.ownerId')
|
||||||
|
.where('album_user.albumId', 'in', (eb) =>
|
||||||
|
eb.selectFrom('album_user').select('album_user.albumId').where('album_user.userId', '=', userId),
|
||||||
|
)
|
||||||
|
.union(
|
||||||
|
eb
|
||||||
|
.selectFrom('partner')
|
||||||
|
.select((eb) => eb.fn<SharingPermission>('unnest', ['partner.permissions']).as('permission'))
|
||||||
|
.distinct()
|
||||||
|
.whereRef('partner.sharedById', '=', 'asset.ownerId')
|
||||||
|
.where('partner.sharedWithId', '=', userId),
|
||||||
|
),
|
||||||
|
).as('permissions');
|
||||||
|
}
|
||||||
|
|
||||||
export function withTagId<O>(qb: SelectQueryBuilder<DB, 'asset', O>, tagId: string) {
|
export function withTagId<O>(qb: SelectQueryBuilder<DB, 'asset', O>, tagId: string) {
|
||||||
return qb.where((eb) =>
|
return qb.where((eb) =>
|
||||||
eb.exists(
|
eb.exists(
|
||||||
@@ -428,7 +464,7 @@ export function searchAssetBuilder(kysely: Kysely<DB>, options: AssetSearchBuild
|
|||||||
.$if(!!options.checksum, (qb) => qb.where('asset.checksum', '=', options.checksum!))
|
.$if(!!options.checksum, (qb) => qb.where('asset.checksum', '=', options.checksum!))
|
||||||
.$if(!!options.id, (qb) => qb.where('asset.id', '=', asUuid(options.id!)))
|
.$if(!!options.id, (qb) => qb.where('asset.id', '=', asUuid(options.id!)))
|
||||||
.$if(!!options.libraryId, (qb) => qb.where('asset.libraryId', '=', asUuid(options.libraryId!)))
|
.$if(!!options.libraryId, (qb) => qb.where('asset.libraryId', '=', asUuid(options.libraryId!)))
|
||||||
.$if(!!options.userIds, (qb) => qb.where('asset.ownerId', '=', anyUuid(options.userIds!)))
|
.$if(!!options.userIds, (qb) => qb.where(hasAssetPermissions(options.userIds![0], [SharingPermission.AssetRead])))
|
||||||
.$if(!!options.encodedVideoPath, (qb) =>
|
.$if(!!options.encodedVideoPath, (qb) =>
|
||||||
qb
|
qb
|
||||||
.innerJoin('asset_file', (join) =>
|
.innerJoin('asset_file', (join) =>
|
||||||
|
|||||||
@@ -38,6 +38,7 @@ const createAsset = (
|
|||||||
fileSizeInByte !== null || Object.keys(exifFields).length > 0
|
fileSizeInByte !== null || Object.keys(exifFields).length > 0
|
||||||
? ExifResponseSchema.parse({ fileSizeInByte, ...exifFields })
|
? ExifResponseSchema.parse({ fileSizeInByte, ...exifFields })
|
||||||
: undefined,
|
: undefined,
|
||||||
|
permissions: [],
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('duplicate utils', () => {
|
describe('duplicate utils', () => {
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import { AssetFace } from 'src/database';
|
|
||||||
import { AssetOcrResponseDto } from 'src/dtos/ocr.dto';
|
import { AssetOcrResponseDto } from 'src/dtos/ocr.dto';
|
||||||
import { ImageDimensions } from 'src/types';
|
import { ImageDimensions } from 'src/types';
|
||||||
|
|
||||||
@@ -31,11 +30,21 @@ const scale = (box: BoundingBox, target: ImageDimensions, source?: ImageDimensio
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export const checkFaceVisibility = (
|
export const checkFaceVisibility = <
|
||||||
faces: AssetFace[],
|
T extends {
|
||||||
|
isVisible: boolean;
|
||||||
|
boundingBoxX1: number;
|
||||||
|
boundingBoxX2: number;
|
||||||
|
boundingBoxY1: number;
|
||||||
|
boundingBoxY2: number;
|
||||||
|
imageHeight: number;
|
||||||
|
imageWidth: number;
|
||||||
|
},
|
||||||
|
>(
|
||||||
|
faces: T[],
|
||||||
originalAssetDimensions: ImageDimensions,
|
originalAssetDimensions: ImageDimensions,
|
||||||
crop?: BoundingBox,
|
crop?: BoundingBox,
|
||||||
): { visible: AssetFace[]; hidden: AssetFace[] } => {
|
): { visible: T[]; hidden: T[] } => {
|
||||||
if (!crop) {
|
if (!crop) {
|
||||||
return {
|
return {
|
||||||
visible: faces.filter((face) => !face.isVisible),
|
visible: faces.filter((face) => !face.isVisible),
|
||||||
|
|||||||
@@ -28,6 +28,8 @@ export class AlbumUserFactory {
|
|||||||
createdAt: newDate(),
|
createdAt: newDate(),
|
||||||
updateId: newUuidV7(),
|
updateId: newUuidV7(),
|
||||||
updatedAt: newDate(),
|
updatedAt: newDate(),
|
||||||
|
permissions: [],
|
||||||
|
inTimeline: false,
|
||||||
...dto,
|
...dto,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ export class PartnerFactory {
|
|||||||
sharedWithId,
|
sharedWithId,
|
||||||
updatedAt: newDate(),
|
updatedAt: newDate(),
|
||||||
updateId: newUuidV7(),
|
updateId: newUuidV7(),
|
||||||
|
permissions: [],
|
||||||
...dto,
|
...dto,
|
||||||
})
|
})
|
||||||
.sharedBy({ id: sharedById })
|
.sharedBy({ id: sharedById })
|
||||||
|
|||||||
@@ -35,6 +35,7 @@ export class UserFactory {
|
|||||||
status: UserStatus.Active,
|
status: UserStatus.Active,
|
||||||
profileChangedAt: newDate(),
|
profileChangedAt: newDate(),
|
||||||
updateId: newUuidV7(),
|
updateId: newUuidV7(),
|
||||||
|
trustedGroupId: newUuid(),
|
||||||
...dto,
|
...dto,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ export const newAccessRepositoryMock = (): IAccessRepositoryMock => {
|
|||||||
checkAlbumAccess: vitest.fn().mockResolvedValue(new Set()),
|
checkAlbumAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
checkPartnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
checkPartnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
checkSharedLinkAccess: vitest.fn().mockResolvedValue(new Set()),
|
checkSharedLinkAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
|
checkSharedAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
},
|
},
|
||||||
|
|
||||||
album: {
|
album: {
|
||||||
@@ -48,6 +49,8 @@ export const newAccessRepositoryMock = (): IAccessRepositoryMock => {
|
|||||||
person: {
|
person: {
|
||||||
checkFaceOwnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
checkFaceOwnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
checkOwnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
checkOwnerAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
|
checkSharedAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
|
checkSharedFaceAccess: vitest.fn().mockResolvedValue(new Set()),
|
||||||
},
|
},
|
||||||
|
|
||||||
partner: {
|
partner: {
|
||||||
|
|||||||
@@ -26,12 +26,13 @@
|
|||||||
import { Route } from '$lib/route';
|
import { Route } from '$lib/route';
|
||||||
import { getGlobalActions } from '$lib/services/app.service';
|
import { getGlobalActions } from '$lib/services/app.service';
|
||||||
import { getAssetActions } from '$lib/services/asset.service';
|
import { getAssetActions } from '$lib/services/asset.service';
|
||||||
import { getSharedLink, withoutIcons } from '$lib/utils';
|
import { getSharedLink, hasPermissions, withoutIcons } from '$lib/utils';
|
||||||
import type { OnUndoDelete } from '$lib/utils/actions';
|
import type { OnUndoDelete } from '$lib/utils/actions';
|
||||||
import { toTimelineAsset } from '$lib/utils/timeline-util';
|
import { toTimelineAsset } from '$lib/utils/timeline-util';
|
||||||
import {
|
import {
|
||||||
AssetTypeEnum,
|
AssetTypeEnum,
|
||||||
AssetVisibility,
|
AssetVisibility,
|
||||||
|
SharingPermission,
|
||||||
type AlbumResponseDto,
|
type AlbumResponseDto,
|
||||||
type AssetResponseDto,
|
type AssetResponseDto,
|
||||||
type PersonResponseDto,
|
type PersonResponseDto,
|
||||||
@@ -141,7 +142,7 @@
|
|||||||
|
|
||||||
<ActionButton action={Actions.Edit} />
|
<ActionButton action={Actions.Edit} />
|
||||||
|
|
||||||
{#if isOwner}
|
{#if hasPermissions(asset, SharingPermission.AssetDelete)}
|
||||||
<DeleteAction {asset} {onAction} {preAction} {onUndoDelete} />
|
<DeleteAction {asset} {onAction} {preAction} {onUndoDelete} />
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
@@ -159,7 +160,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<ActionMenuItem action={Actions.AddToAlbum} />
|
<ActionMenuItem action={Actions.AddToAlbum} />
|
||||||
{#if album && (isOwner || isAlbumOwner)}
|
{#if album && (hasPermissions(asset, SharingPermission.AssetShare) || isAlbumOwner)}
|
||||||
<RemoveFromAlbumAction {album} onRemove={onRemoveFromAlbum} assetIds={[asset.id]} menuItem />
|
<RemoveFromAlbumAction {album} onRemove={onRemoveFromAlbum} assetIds={[asset.id]} menuItem />
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
@@ -187,7 +188,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
{#if !isLocked}
|
{#if !isLocked}
|
||||||
{#if isOwner}
|
{#if hasPermissions(asset, SharingPermission.AssetUpdate)}
|
||||||
<ArchiveAction {asset} {onAction} {preAction} />
|
<ArchiveAction {asset} {onAction} {preAction} />
|
||||||
{#if !asset.isArchived && !asset.isTrashed}
|
{#if !asset.isArchived && !asset.isTrashed}
|
||||||
<MenuOption
|
<MenuOption
|
||||||
@@ -217,7 +218,7 @@
|
|||||||
text={playOriginalVideo ? $t('play_transcoded_video') : $t('play_original_video')}
|
text={playOriginalVideo ? $t('play_transcoded_video') : $t('play_original_video')}
|
||||||
/>
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
{#if isOwner}
|
{#if hasPermissions(asset, SharingPermission.AssetUpdate)}
|
||||||
<hr />
|
<hr />
|
||||||
<ActionMenuItem action={Actions.RefreshFacesJob} />
|
<ActionMenuItem action={Actions.RefreshFacesJob} />
|
||||||
<ActionMenuItem action={Actions.RefreshMetadataJob} />
|
<ActionMenuItem action={Actions.RefreshMetadataJob} />
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
import DetailPanelDate from '$lib/components/asset-viewer/DetailPanelDate.svelte';
|
import DetailPanelDate from '$lib/components/asset-viewer/DetailPanelDate.svelte';
|
||||||
import DetailPanelDescription from '$lib/components/asset-viewer/DetailPanelDescription.svelte';
|
import DetailPanelDescription from '$lib/components/asset-viewer/DetailPanelDescription.svelte';
|
||||||
import DetailPanelLocation from '$lib/components/asset-viewer/DetailPanelLocation.svelte';
|
import DetailPanelLocation from '$lib/components/asset-viewer/DetailPanelLocation.svelte';
|
||||||
|
import DetailPanelPeople from '$lib/components/asset-viewer/DetailPanelPeople.svelte';
|
||||||
import DetailPanelRating from '$lib/components/asset-viewer/DetailPanelStarRating.svelte';
|
import DetailPanelRating from '$lib/components/asset-viewer/DetailPanelStarRating.svelte';
|
||||||
import DetailPanelTags from '$lib/components/asset-viewer/DetailPanelTags.svelte';
|
import DetailPanelTags from '$lib/components/asset-viewer/DetailPanelTags.svelte';
|
||||||
import { timeToLoadTheMap } from '$lib/constants';
|
import { timeToLoadTheMap } from '$lib/constants';
|
||||||
@@ -11,7 +12,7 @@
|
|||||||
import { featureFlagsManager } from '$lib/managers/feature-flags-manager.svelte';
|
import { featureFlagsManager } from '$lib/managers/feature-flags-manager.svelte';
|
||||||
import { Route } from '$lib/route';
|
import { Route } from '$lib/route';
|
||||||
import { locale } from '$lib/stores/preferences.store';
|
import { locale } from '$lib/stores/preferences.store';
|
||||||
import { getAssetMediaUrl } from '$lib/utils';
|
import { getAssetMediaUrl, hasPermissions } from '$lib/utils';
|
||||||
import { delay, getDimensions } from '$lib/utils/asset-utils';
|
import { delay, getDimensions } from '$lib/utils/asset-utils';
|
||||||
import { getByteUnitString } from '$lib/utils/byte-units';
|
import { getByteUnitString } from '$lib/utils/byte-units';
|
||||||
import { handleError } from '$lib/utils/handle-error';
|
import { handleError } from '$lib/utils/handle-error';
|
||||||
@@ -20,6 +21,7 @@
|
|||||||
AssetMediaSize,
|
AssetMediaSize,
|
||||||
getAllAlbums,
|
getAllAlbums,
|
||||||
getAssetInfo,
|
getAssetInfo,
|
||||||
|
SharingPermission,
|
||||||
type AlbumResponseDto,
|
type AlbumResponseDto,
|
||||||
type AssetResponseDto,
|
type AssetResponseDto,
|
||||||
} from '@immich/sdk';
|
} from '@immich/sdk';
|
||||||
@@ -32,7 +34,6 @@
|
|||||||
import OnEvents from '../OnEvents.svelte';
|
import OnEvents from '../OnEvents.svelte';
|
||||||
import UserAvatar from '../shared-components/UserAvatar.svelte';
|
import UserAvatar from '../shared-components/UserAvatar.svelte';
|
||||||
import AlbumListItemDetails from './AlbumListItemDetails.svelte';
|
import AlbumListItemDetails from './AlbumListItemDetails.svelte';
|
||||||
import DetailPanelPeople from '$lib/components/asset-viewer/DetailPanelPeople.svelte';
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
@@ -42,6 +43,7 @@
|
|||||||
let { asset, currentAlbum = null }: Props = $props();
|
let { asset, currentAlbum = null }: Props = $props();
|
||||||
|
|
||||||
let isOwner = $derived(authManager.authenticated && authManager.user.id === asset.ownerId);
|
let isOwner = $derived(authManager.authenticated && authManager.user.id === asset.ownerId);
|
||||||
|
const allowExifUpdate = $derived(hasPermissions(asset, SharingPermission.AssetUpdate, SharingPermission.ExifRead));
|
||||||
let latlng = $derived(
|
let latlng = $derived(
|
||||||
(() => {
|
(() => {
|
||||||
const lat = asset.exifInfo?.latitude;
|
const lat = asset.exifInfo?.latitude;
|
||||||
@@ -147,9 +149,9 @@
|
|||||||
</section>
|
</section>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<DetailPanelDescription {asset} {isOwner} />
|
<DetailPanelDescription {asset} {allowExifUpdate} />
|
||||||
<DetailPanelRating {asset} {isOwner} />
|
<DetailPanelRating {asset} {allowExifUpdate} />
|
||||||
<DetailPanelPeople {asset} {isOwner} {previousRoute} />
|
<DetailPanelPeople {asset} {previousRoute} />
|
||||||
|
|
||||||
<div class="p-4">
|
<div class="p-4">
|
||||||
{#if asset.exifInfo}
|
{#if asset.exifInfo}
|
||||||
@@ -160,7 +162,7 @@
|
|||||||
<Text size="small" color="muted">{$t('no_exif_info_available')}</Text>
|
<Text size="small" color="muted">{$t('no_exif_info_available')}</Text>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<DetailPanelDate {asset} />
|
<DetailPanelDate {asset} {allowExifUpdate} />
|
||||||
|
|
||||||
<div class="flex gap-4 py-4">
|
<div class="flex gap-4 py-4">
|
||||||
<div><Icon icon={mdiImageOutline} size="24" /></div>
|
<div><Icon icon={mdiImageOutline} size="24" /></div>
|
||||||
@@ -168,7 +170,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<p class="flex place-items-center gap-2 break-all whitespace-pre-wrap">
|
<p class="flex place-items-center gap-2 break-all whitespace-pre-wrap">
|
||||||
{asset.originalFileName}
|
{asset.originalFileName}
|
||||||
{#if isOwner}
|
{#if allowExifUpdate}
|
||||||
<IconButton
|
<IconButton
|
||||||
icon={mdiInformationOutline}
|
icon={mdiInformationOutline}
|
||||||
aria-label={$t('show_file_location')}
|
aria-label={$t('show_file_location')}
|
||||||
@@ -271,7 +273,7 @@
|
|||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<DetailPanelLocation {isOwner} {asset} />
|
<DetailPanelLocation {allowExifUpdate} {asset} />
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
|||||||
@@ -10,9 +10,10 @@
|
|||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
|
allowExifUpdate: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
const { asset }: Props = $props();
|
const { asset, allowExifUpdate }: Props = $props();
|
||||||
|
|
||||||
const timeZone = $derived(asset.exifInfo?.timeZone ?? undefined);
|
const timeZone = $derived(asset.exifInfo?.timeZone ?? undefined);
|
||||||
const dateTime = $derived(
|
const dateTime = $derived(
|
||||||
@@ -20,13 +21,8 @@
|
|||||||
? fromISODateTime(asset.exifInfo.dateTimeOriginal, timeZone)
|
? fromISODateTime(asset.exifInfo.dateTimeOriginal, timeZone)
|
||||||
: fromISODateTimeUTC(asset.localDateTime),
|
: fromISODateTimeUTC(asset.localDateTime),
|
||||||
);
|
);
|
||||||
const isOwner = $derived(authManager.authenticated && asset.ownerId === authManager.user.id);
|
|
||||||
|
|
||||||
const handleChangeDate = async () => {
|
const handleChangeDate = async () => {
|
||||||
if (!isOwner) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await modalManager.show(AssetChangeDateModal, {
|
await modalManager.show(AssetChangeDateModal, {
|
||||||
asset: toTimelineAsset(asset),
|
asset: toTimelineAsset(asset),
|
||||||
initialDate: dateTime,
|
initialDate: dateTime,
|
||||||
@@ -40,8 +36,8 @@
|
|||||||
type="button"
|
type="button"
|
||||||
class="flex w-full place-items-start justify-between gap-4 py-4 text-start"
|
class="flex w-full place-items-start justify-between gap-4 py-4 text-start"
|
||||||
onclick={handleChangeDate}
|
onclick={handleChangeDate}
|
||||||
title={isOwner ? $t('edit_date') : ''}
|
title={allowExifUpdate ? $t('edit_date') : ''}
|
||||||
class:hover:text-primary={isOwner}
|
class:hover:text-primary={allowExifUpdate}
|
||||||
data-testid="detail-panel-edit-date-button"
|
data-testid="detail-panel-edit-date-button"
|
||||||
>
|
>
|
||||||
<div class="flex gap-4">
|
<div class="flex gap-4">
|
||||||
@@ -68,13 +64,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if isOwner}
|
{#if allowExifUpdate}
|
||||||
<div class="p-1">
|
<div class="p-1">
|
||||||
<Icon icon={mdiPencil} size="20" />
|
<Icon icon={mdiPencil} size="20" />
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
</button>
|
</button>
|
||||||
{:else if !dateTime && isOwner}
|
{:else if !dateTime && allowExifUpdate}
|
||||||
<div class="flex place-items-start justify-between gap-4 py-4">
|
<div class="flex place-items-start justify-between gap-4 py-4">
|
||||||
<div class="flex gap-4">
|
<div class="flex gap-4">
|
||||||
<Icon icon={mdiCalendar} size="24" />
|
<Icon icon={mdiCalendar} size="24" />
|
||||||
|
|||||||
@@ -8,10 +8,10 @@
|
|||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
isOwner: boolean;
|
allowExifUpdate: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
let { asset, isOwner }: Props = $props();
|
let { asset, allowExifUpdate }: Props = $props();
|
||||||
|
|
||||||
let description = $derived(asset.exifInfo?.description ?? '');
|
let description = $derived(asset.exifInfo?.description ?? '');
|
||||||
|
|
||||||
@@ -29,7 +29,7 @@
|
|||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if isOwner}
|
{#if allowExifUpdate}
|
||||||
<section class="mt-10 px-4">
|
<section class="mt-10 px-4">
|
||||||
<Textarea
|
<Textarea
|
||||||
bind:value={description}
|
bind:value={description}
|
||||||
|
|||||||
@@ -7,11 +7,11 @@
|
|||||||
import { t } from 'svelte-i18n';
|
import { t } from 'svelte-i18n';
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
isOwner: boolean;
|
allowExifUpdate: boolean;
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
};
|
};
|
||||||
|
|
||||||
let { isOwner, asset = $bindable() }: Props = $props();
|
let { allowExifUpdate, asset = $bindable() }: Props = $props();
|
||||||
|
|
||||||
const onAction = async () => {
|
const onAction = async () => {
|
||||||
const point = await modalManager.show(GeolocationPointPickerModal, { asset });
|
const point = await modalManager.show(GeolocationPointPickerModal, { asset });
|
||||||
@@ -34,9 +34,9 @@
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="flex w-full place-items-start justify-between gap-4 py-4 text-start"
|
class="flex w-full place-items-start justify-between gap-4 py-4 text-start"
|
||||||
onclick={isOwner ? onAction : undefined}
|
onclick={allowExifUpdate ? onAction : undefined}
|
||||||
title={isOwner ? $t('edit_location') : ''}
|
title={allowExifUpdate ? $t('edit_location') : ''}
|
||||||
class:hover:text-primary={isOwner}
|
class:hover:text-primary={allowExifUpdate}
|
||||||
>
|
>
|
||||||
<div class="flex gap-4">
|
<div class="flex gap-4">
|
||||||
<div><Icon icon={mdiMapMarkerOutline} size="24" /></div>
|
<div><Icon icon={mdiMapMarkerOutline} size="24" /></div>
|
||||||
@@ -58,13 +58,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if isOwner}
|
{#if allowExifUpdate}
|
||||||
<div>
|
<div>
|
||||||
<Icon icon={mdiPencil} size="20" />
|
<Icon icon={mdiPencil} size="20" />
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
</button>
|
</button>
|
||||||
{:else if !asset.exifInfo?.city && isOwner}
|
{:else if !asset.exifInfo?.city && allowExifUpdate}
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="flex w-full place-items-start justify-between gap-4 rounded-lg py-4 text-start hover:text-primary"
|
class="flex w-full place-items-start justify-between gap-4 rounded-lg py-4 text-start hover:text-primary"
|
||||||
|
|||||||
@@ -5,8 +5,8 @@
|
|||||||
import { Route } from '$lib/route';
|
import { Route } from '$lib/route';
|
||||||
import { faceManager } from '$lib/stores/face.svelte';
|
import { faceManager } from '$lib/stores/face.svelte';
|
||||||
import { locale } from '$lib/stores/preferences.store';
|
import { locale } from '$lib/stores/preferences.store';
|
||||||
import { getPeopleThumbnailUrl } from '$lib/utils';
|
import { getPeopleThumbnailUrl, hasPermissions } from '$lib/utils';
|
||||||
import { type AssetResponseDto } from '@immich/sdk';
|
import { SharingPermission, type AssetResponseDto } from '@immich/sdk';
|
||||||
import { IconButton, Text } from '@immich/ui';
|
import { IconButton, Text } from '@immich/ui';
|
||||||
import { mdiEye, mdiEyeOff, mdiPencil, mdiPlus } from '@mdi/js';
|
import { mdiEye, mdiEyeOff, mdiPencil, mdiPlus } from '@mdi/js';
|
||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
@@ -14,13 +14,13 @@
|
|||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
isOwner: boolean;
|
|
||||||
previousRoute: string;
|
previousRoute: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
const { asset, isOwner, previousRoute }: Props = $props();
|
const { asset, previousRoute }: Props = $props();
|
||||||
|
|
||||||
const people = $derived(Array.from(faceManager.people));
|
const people = $derived(Array.from(faceManager.people));
|
||||||
|
$effect(() => console.log(people));
|
||||||
const visiblePeople = $derived(
|
const visiblePeople = $derived(
|
||||||
people
|
people
|
||||||
.filter((p) => assetViewerManager.isShowingHiddenPeople || !p.isHidden)
|
.filter((p) => assetViewerManager.isShowingHiddenPeople || !p.isHidden)
|
||||||
@@ -56,7 +56,7 @@
|
|||||||
);
|
);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if !authManager.isSharedLink && isOwner}
|
{#if !authManager.isSharedLink && hasPermissions(asset, SharingPermission.PersonRead)}
|
||||||
<section class="px-4 pt-4 text-sm">
|
<section class="px-4 pt-4 text-sm">
|
||||||
<div class="flex h-10 w-full items-center justify-between">
|
<div class="flex h-10 w-full items-center justify-between">
|
||||||
<Text size="small" color="muted">{$t('people')}</Text>
|
<Text size="small" color="muted">{$t('people')}</Text>
|
||||||
|
|||||||
@@ -8,10 +8,10 @@
|
|||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
asset: AssetResponseDto;
|
asset: AssetResponseDto;
|
||||||
isOwner: boolean;
|
allowExifUpdate: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
let { asset, isOwner }: Props = $props();
|
let { asset, allowExifUpdate }: Props = $props();
|
||||||
|
|
||||||
let rating = $derived(asset.exifInfo?.rating || null) as Rating;
|
let rating = $derived(asset.exifInfo?.rating || null) as Rating;
|
||||||
|
|
||||||
@@ -26,6 +26,10 @@
|
|||||||
|
|
||||||
{#if !authManager.isSharedLink && authManager.authenticated && authManager.preferences.ratings.enabled}
|
{#if !authManager.isSharedLink && authManager.authenticated && authManager.preferences.ratings.enabled}
|
||||||
<section class="px-4 pt-4">
|
<section class="px-4 pt-4">
|
||||||
<StarRating {rating} readOnly={!isOwner} onRating={(rating) => handlePromiseError(handleChangeRating(rating))} />
|
<StarRating
|
||||||
|
{rating}
|
||||||
|
readOnly={!allowExifUpdate}
|
||||||
|
onRating={(rating) => handlePromiseError(handleChangeRating(rating))}
|
||||||
|
/>
|
||||||
</section>
|
</section>
|
||||||
{/if}
|
{/if}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@
|
|||||||
{ title: $t('admin.memory_cleanup_job'), value: ManualJobName.MemoryCleanup },
|
{ title: $t('admin.memory_cleanup_job'), value: ManualJobName.MemoryCleanup },
|
||||||
{ title: $t('admin.memory_generate_job'), value: ManualJobName.MemoryCreate },
|
{ title: $t('admin.memory_generate_job'), value: ManualJobName.MemoryCreate },
|
||||||
{ title: $t('admin.backup_database'), value: ManualJobName.BackupDatabase },
|
{ title: $t('admin.backup_database'), value: ManualJobName.BackupDatabase },
|
||||||
|
{ title: 'Person grouping', value: ManualJobName.PersonGroupMerge },
|
||||||
].map(({ value, title }) => ({ id: value, label: title, value }));
|
].map(({ value, title }) => ({ id: value, label: title, value }));
|
||||||
|
|
||||||
let selectedJob: ComboBoxOption | undefined = $state(undefined);
|
let selectedJob: ComboBoxOption | undefined = $state(undefined);
|
||||||
|
|||||||
@@ -36,7 +36,7 @@
|
|||||||
try {
|
try {
|
||||||
await mergePerson({
|
await mergePerson({
|
||||||
id: personToBeMergedInto.id,
|
id: personToBeMergedInto.id,
|
||||||
mergePersonDto: { ids: [personToMerge.id] },
|
mergeFaceClusterDto: { ids: [personToMerge.id] },
|
||||||
});
|
});
|
||||||
toastManager.primary($t('merge_people_successfully'));
|
toastManager.primary($t('merge_people_successfully'));
|
||||||
onClose([personToMerge, personToBeMergedInto]);
|
onClose([personToMerge, personToBeMergedInto]);
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { getOwnAlbumUser, SharingPermission, updateOwnAlbumUser } from '@immich/sdk';
|
||||||
|
import { Checkbox, Field, FormModal, Heading, Stack, Switch, toastManager } from '@immich/ui';
|
||||||
|
import { onMount } from 'svelte';
|
||||||
|
import { init } from 'svelte-i18n';
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
onClose: () => void;
|
||||||
|
albumId?: string;
|
||||||
|
partnerId?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const { onClose, ...rest }: Props = $props();
|
||||||
|
|
||||||
|
let checkedPermissions = $state<SharingPermission[]>([]);
|
||||||
|
let viewInTimeline = $state<boolean>(false);
|
||||||
|
|
||||||
|
const onCheckedChange = (permission: SharingPermission, checked: boolean) => {
|
||||||
|
if (checked) {
|
||||||
|
checkedPermissions.push(permission);
|
||||||
|
} else {
|
||||||
|
checkedPermissions = checkedPermissions.filter((perm) => perm !== permission);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onSubmit = async () => {
|
||||||
|
const permissions =
|
||||||
|
checkedPermissions.length === Object.values(SharingPermission).length - 1
|
||||||
|
? [SharingPermission.All]
|
||||||
|
: checkedPermissions;
|
||||||
|
if (rest.albumId) {
|
||||||
|
await updateOwnAlbumUser({
|
||||||
|
id: rest.albumId,
|
||||||
|
updateSharingOptionsDto: { permissions, inTimeline: viewInTimeline },
|
||||||
|
});
|
||||||
|
toastManager.success();
|
||||||
|
}
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
onMount(async () => {
|
||||||
|
if (rest.albumId) {
|
||||||
|
const { permissions, inTimeline } = await getOwnAlbumUser({ id: rest.albumId });
|
||||||
|
checkedPermissions = permissions;
|
||||||
|
viewInTimeline = inTimeline;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormModal title="Sharing options" {onClose} {onSubmit}>
|
||||||
|
<Stack>
|
||||||
|
<Field label="View in timeline">
|
||||||
|
<Switch bind:checked={viewInTimeline} />
|
||||||
|
</Field>
|
||||||
|
<Heading>Permissions</Heading>
|
||||||
|
<Field label={SharingPermission.All}>
|
||||||
|
<Checkbox
|
||||||
|
id="permission-{SharingPermission.All}"
|
||||||
|
checked={checkedPermissions.length === Object.values(SharingPermission).length - 1}
|
||||||
|
onCheckedChange={(checked) =>
|
||||||
|
checked
|
||||||
|
? (checkedPermissions = Object.values(SharingPermission).filter(
|
||||||
|
(permission) => permission !== SharingPermission.All,
|
||||||
|
))
|
||||||
|
: (checkedPermissions = [])}
|
||||||
|
/>
|
||||||
|
</Field>
|
||||||
|
{#each Object.values(SharingPermission).filter((permission) => permission !== SharingPermission.All) as permission (permission)}
|
||||||
|
<Field label={permission}>
|
||||||
|
<Checkbox
|
||||||
|
id="permission-{permission}"
|
||||||
|
checked={checkedPermissions.includes(permission)}
|
||||||
|
onCheckedChange={(checked) => onCheckedChange(permission, checked)}
|
||||||
|
/>
|
||||||
|
</Field>
|
||||||
|
{/each}
|
||||||
|
</Stack>
|
||||||
|
</FormModal>
|
||||||
@@ -5,6 +5,7 @@ import {
|
|||||||
AssetVisibility,
|
AssetVisibility,
|
||||||
getAssetInfo,
|
getAssetInfo,
|
||||||
runAssetJobs,
|
runAssetJobs,
|
||||||
|
SharingPermission,
|
||||||
updateAsset,
|
updateAsset,
|
||||||
type AssetJobsDto,
|
type AssetJobsDto,
|
||||||
type AssetResponseDto,
|
type AssetResponseDto,
|
||||||
@@ -41,7 +42,7 @@ import { eventManager } from '$lib/managers/event-manager.svelte';
|
|||||||
import AssetAddToAlbumModal from '$lib/modals/AssetAddToAlbumModal.svelte';
|
import AssetAddToAlbumModal from '$lib/modals/AssetAddToAlbumModal.svelte';
|
||||||
import AssetTagModal from '$lib/modals/AssetTagModal.svelte';
|
import AssetTagModal from '$lib/modals/AssetTagModal.svelte';
|
||||||
import SharedLinkCreateModal from '$lib/modals/SharedLinkCreateModal.svelte';
|
import SharedLinkCreateModal from '$lib/modals/SharedLinkCreateModal.svelte';
|
||||||
import { getAssetMediaUrl, getSharedLink, sleep } from '$lib/utils';
|
import { getAssetMediaUrl, getSharedLink, hasPermissions, sleep } from '$lib/utils';
|
||||||
import { downloadUrl } from '$lib/utils/asset-utils';
|
import { downloadUrl } from '$lib/utils/asset-utils';
|
||||||
import { handleError } from '$lib/utils/handle-error';
|
import { handleError } from '$lib/utils/handle-error';
|
||||||
import { getFormatter } from '$lib/utils/i18n';
|
import { getFormatter } from '$lib/utils/i18n';
|
||||||
@@ -98,7 +99,12 @@ export const getAssetActions = ($t: MessageFormatter, asset: AssetResponseDto) =
|
|||||||
const Share: ActionItem = {
|
const Share: ActionItem = {
|
||||||
title: $t('share'),
|
title: $t('share'),
|
||||||
icon: mdiShareVariantOutline,
|
icon: mdiShareVariantOutline,
|
||||||
$if: () => !!(authUser && !asset.isTrashed && asset.visibility !== AssetVisibility.Locked),
|
$if: () =>
|
||||||
|
!!(
|
||||||
|
hasPermissions(asset, SharingPermission.AssetShare) &&
|
||||||
|
!asset.isTrashed &&
|
||||||
|
asset.visibility !== AssetVisibility.Locked
|
||||||
|
),
|
||||||
onAction: () => modalManager.show(SharedLinkCreateModal, { assetIds: [asset.id] }),
|
onAction: () => modalManager.show(SharedLinkCreateModal, { assetIds: [asset.id] }),
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -119,7 +125,7 @@ export const getAssetActions = ($t: MessageFormatter, asset: AssetResponseDto) =
|
|||||||
|
|
||||||
const SharedLinkDownload: ActionItem = {
|
const SharedLinkDownload: ActionItem = {
|
||||||
...Download,
|
...Download,
|
||||||
$if: () => isOwner || !!sharedLink?.allowDownload,
|
$if: () => hasPermissions(asset, SharingPermission.AssetShare) || !!sharedLink?.allowDownload,
|
||||||
};
|
};
|
||||||
|
|
||||||
const PlayMotionPhoto: ActionItem = {
|
const PlayMotionPhoto: ActionItem = {
|
||||||
@@ -222,7 +228,7 @@ export const getAssetActions = ($t: MessageFormatter, asset: AssetResponseDto) =
|
|||||||
icon: mdiTune,
|
icon: mdiTune,
|
||||||
$if: () =>
|
$if: () =>
|
||||||
!sharedLink &&
|
!sharedLink &&
|
||||||
isOwner &&
|
hasPermissions(asset, SharingPermission.AssetEdit) &&
|
||||||
asset.type === AssetTypeEnum.Image &&
|
asset.type === AssetTypeEnum.Image &&
|
||||||
!asset.livePhotoVideoId &&
|
!asset.livePhotoVideoId &&
|
||||||
asset.exifInfo?.projectionType !== ProjectionType.EQUIRECTANGULAR &&
|
asset.exifInfo?.projectionType !== ProjectionType.EQUIRECTANGULAR &&
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import {
|
|||||||
AssetMediaSize,
|
AssetMediaSize,
|
||||||
AssetTypeEnum,
|
AssetTypeEnum,
|
||||||
MemoryType,
|
MemoryType,
|
||||||
|
SharingPermission,
|
||||||
finishOAuth,
|
finishOAuth,
|
||||||
getAssetOriginalPath,
|
getAssetOriginalPath,
|
||||||
getAssetPlaybackPath,
|
getAssetPlaybackPath,
|
||||||
@@ -413,3 +414,17 @@ export const transformToTitleCase = (text: string) => {
|
|||||||
}
|
}
|
||||||
return result.trim();
|
return result.trim();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const hasPermissions = (asset: AssetResponseDto, ...permissions: SharingPermission[]) => {
|
||||||
|
if (asset.permissions.includes(SharingPermission.All)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const permission of permissions) {
|
||||||
|
if (!asset.permissions.includes(permission)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|||||||
+12
-3
@@ -73,11 +73,13 @@
|
|||||||
mdiLink,
|
mdiLink,
|
||||||
mdiPlus,
|
mdiPlus,
|
||||||
mdiPresentationPlay,
|
mdiPresentationPlay,
|
||||||
|
mdiShareVariant,
|
||||||
} from '@mdi/js';
|
} from '@mdi/js';
|
||||||
import { onDestroy } from 'svelte';
|
import { onDestroy } from 'svelte';
|
||||||
import { t } from 'svelte-i18n';
|
import { t } from 'svelte-i18n';
|
||||||
import { fly } from 'svelte/transition';
|
import { fly } from 'svelte/transition';
|
||||||
import type { PageData } from './$types';
|
import type { PageData } from './$types';
|
||||||
|
import SharingOptionsModal from '$lib/modals/SharingOptionsModal.svelte';
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
data: PageData;
|
data: PageData;
|
||||||
@@ -405,9 +407,16 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
{#if isOwned}
|
<IconButton
|
||||||
<ActionButton action={Share} />
|
shape="round"
|
||||||
{/if}
|
aria-label="Sharing permissions"
|
||||||
|
color="secondary"
|
||||||
|
size="medium"
|
||||||
|
icon={mdiShareVariant}
|
||||||
|
onclick={() => modalManager.show(SharingOptionsModal, { albumId: album.id })}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ActionButton action={Share} />
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
<AlbumDescription
|
<AlbumDescription
|
||||||
|
|||||||
@@ -231,7 +231,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
const personWithSimilarName = await findPeopleWithSimilarName(name, targetPerson.id);
|
const personWithSimilarName = await findPeopleWithSimilarName(name, targetPerson.id);
|
||||||
if (personWithSimilarName) {
|
if (personWithSimilarName && personWithSimilarName.faceClusterId !== targetPerson.faceClusterId) {
|
||||||
personMerge1 = targetPerson;
|
personMerge1 = targetPerson;
|
||||||
personMerge2 = personWithSimilarName;
|
personMerge2 = personWithSimilarName;
|
||||||
potentialMergePeople = people
|
potentialMergePeople = people
|
||||||
|
|||||||
+1
-1
@@ -68,7 +68,7 @@
|
|||||||
try {
|
try {
|
||||||
let results = await mergePerson({
|
let results = await mergePerson({
|
||||||
id: person.id,
|
id: person.id,
|
||||||
mergePersonDto: { ids: selectedPeople.map(({ id }) => id) },
|
mergeFaceClusterDto: { ids: selectedPeople.map(({ id }) => id) },
|
||||||
});
|
});
|
||||||
const mergedPerson = await getPerson({ id: person.id });
|
const mergedPerson = await getPerson({ id: person.id });
|
||||||
const count = results.filter(({ success }) => success).length;
|
const count = results.filter(({ success }) => success).length;
|
||||||
|
|||||||
Reference in New Issue
Block a user