mirror of
https://github.com/immich-app/immich.git
synced 2025-10-23 23:12:06 -04:00
feat(server,web) Semantic import path validation (#7076)
* add library validation api * chore: open api * show warning i UI * add flex row * fix e2e * tests * fix tests * enforce path validation * enforce validation on refresh * return 400 on bad import path * add limits to import paths * set response code to 200 * fix e2e * fix lint * fix test * restore e2e folder * fix import * use startsWith * icon color * notify user of failed validation * add parent div to validation * add docs to the import validation * improve library troubleshooting docs * fix button alignment --------- Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
This commit is contained in:
parent
e7a875eadd
commit
b3c7bebbd4
@ -42,23 +42,24 @@ Finally, files can be deleted from Immich via the `Remove Offline Files` job. Th
|
|||||||
|
|
||||||
### Import Paths
|
### Import Paths
|
||||||
|
|
||||||
External libraries use import paths to determine which files to scan. Each library can have multiple import paths so that files from different locations can be added to the same library. Import paths are scanned recursively, and if a file is in multiple import paths, it will only be added once. If the import paths are edited in a way that an external file is no longer in any import path, it will be removed from the library in the same way a deleted file would. If the file is moved back to an import path, it will be added again as if it was a new file.
|
External libraries use import paths to determine which files to scan. Each library can have multiple import paths so that files from different locations can be added to the same library. Import paths are scanned recursively, and if a file is in multiple import paths, it will only be added once. Each import file must be a readable directory that exists on the filesystem; the import path dialog will alert you of any paths that are not accessible.
|
||||||
|
|
||||||
|
If the import paths are edited in a way that an external file is no longer in any import path, it will be removed from the library in the same way a deleted file would. If the file is moved back to an import path, it will be added again as if it was a new file.
|
||||||
|
|
||||||
### Troubleshooting
|
### Troubleshooting
|
||||||
|
|
||||||
Sometimes, an external library will not scan correctly. This can happen if the immich_server or immich_microservices can't access the files. Here are some things to check:
|
Sometimes, an external library will not scan correctly. This can happen if immich_server or immich_microservices can't access the files. Here are some things to check:
|
||||||
|
|
||||||
- Is the external path set correctly?
|
- Is the external path set correctly? Each import path must be contained in the external path.
|
||||||
|
- Make sure the external path does not contain spaces
|
||||||
- In the docker-compose file, are the volumes mounted correctly?
|
- In the docker-compose file, are the volumes mounted correctly?
|
||||||
- Are the volumes identical between the `server` and `microservices` container?
|
- Are the volumes identical between the `server` and `microservices` container?
|
||||||
- Are the import paths set correctly, and do they match the path set in docker-compose file?
|
- Are the import paths set correctly, and do they match the path set in docker-compose file?
|
||||||
- Are you using symbolic link in your import library?
|
- Make sure you don't use symlinks in your import libraries, and that you aren't linking across docker mounts.
|
||||||
- Are the permissions set correctly?
|
- Are the permissions set correctly?
|
||||||
- Are you using forward slashes everywhere? (`/`)
|
- Make sure you are using forward slashes (`/`) and not backward slashes.
|
||||||
- Are you using symlink across docker mounts?
|
|
||||||
- Are you using [spaces in the internal path](/docs/features/libraries#:~:text=can%20be%20accessed.-,NOTE,-Spaces%20in%20the)?
|
|
||||||
|
|
||||||
If all else fails, you can always start a shell inside the container and check if the path is accessible. For example, `docker exec -it immich_microservices /bin/bash` will start a bash shell. If your import path, for instance, is `/data/import/photos`, you can check if the files are accessible by running `ls /data/import/photos`. Also check the `immich_server` container in the same way.
|
To validate that Immich can reach your external library, start a shell inside the container. Run `docker exec -it immich_microservices /bin/bash` to a bash shell. If your import path is `/data/import/photos`, check it with `ls /data/import/photos`. Do the same check for the `immich_server` container. If you cannot access this directory in both the `microservices` and `server` containers, Immich won't be able to import files.
|
||||||
|
|
||||||
### Security Considerations
|
### Security Considerations
|
||||||
|
|
||||||
|
9
mobile/openapi/.openapi-generator/FILES
generated
9
mobile/openapi/.openapi-generator/FILES
generated
@ -182,6 +182,9 @@ doc/UserAvatarColor.md
|
|||||||
doc/UserDto.md
|
doc/UserDto.md
|
||||||
doc/UserResponseDto.md
|
doc/UserResponseDto.md
|
||||||
doc/ValidateAccessTokenResponseDto.md
|
doc/ValidateAccessTokenResponseDto.md
|
||||||
|
doc/ValidateLibraryDto.md
|
||||||
|
doc/ValidateLibraryImportPathResponseDto.md
|
||||||
|
doc/ValidateLibraryResponseDto.md
|
||||||
doc/VideoCodec.md
|
doc/VideoCodec.md
|
||||||
git_push.sh
|
git_push.sh
|
||||||
lib/api.dart
|
lib/api.dart
|
||||||
@ -372,6 +375,9 @@ lib/model/user_avatar_color.dart
|
|||||||
lib/model/user_dto.dart
|
lib/model/user_dto.dart
|
||||||
lib/model/user_response_dto.dart
|
lib/model/user_response_dto.dart
|
||||||
lib/model/validate_access_token_response_dto.dart
|
lib/model/validate_access_token_response_dto.dart
|
||||||
|
lib/model/validate_library_dto.dart
|
||||||
|
lib/model/validate_library_import_path_response_dto.dart
|
||||||
|
lib/model/validate_library_response_dto.dart
|
||||||
lib/model/video_codec.dart
|
lib/model/video_codec.dart
|
||||||
pubspec.yaml
|
pubspec.yaml
|
||||||
test/activity_api_test.dart
|
test/activity_api_test.dart
|
||||||
@ -553,4 +559,7 @@ test/user_avatar_color_test.dart
|
|||||||
test/user_dto_test.dart
|
test/user_dto_test.dart
|
||||||
test/user_response_dto_test.dart
|
test/user_response_dto_test.dart
|
||||||
test/validate_access_token_response_dto_test.dart
|
test/validate_access_token_response_dto_test.dart
|
||||||
|
test/validate_library_dto_test.dart
|
||||||
|
test/validate_library_import_path_response_dto_test.dart
|
||||||
|
test/validate_library_response_dto_test.dart
|
||||||
test/video_codec_test.dart
|
test/video_codec_test.dart
|
||||||
|
4
mobile/openapi/README.md
generated
4
mobile/openapi/README.md
generated
@ -141,6 +141,7 @@ Class | Method | HTTP request | Description
|
|||||||
*LibraryApi* | [**removeOfflineFiles**](doc//LibraryApi.md#removeofflinefiles) | **POST** /library/{id}/removeOffline |
|
*LibraryApi* | [**removeOfflineFiles**](doc//LibraryApi.md#removeofflinefiles) | **POST** /library/{id}/removeOffline |
|
||||||
*LibraryApi* | [**scanLibrary**](doc//LibraryApi.md#scanlibrary) | **POST** /library/{id}/scan |
|
*LibraryApi* | [**scanLibrary**](doc//LibraryApi.md#scanlibrary) | **POST** /library/{id}/scan |
|
||||||
*LibraryApi* | [**updateLibrary**](doc//LibraryApi.md#updatelibrary) | **PUT** /library/{id} |
|
*LibraryApi* | [**updateLibrary**](doc//LibraryApi.md#updatelibrary) | **PUT** /library/{id} |
|
||||||
|
*LibraryApi* | [**validate**](doc//LibraryApi.md#validate) | **POST** /library/{id}/validate |
|
||||||
*OAuthApi* | [**finishOAuth**](doc//OAuthApi.md#finishoauth) | **POST** /oauth/callback |
|
*OAuthApi* | [**finishOAuth**](doc//OAuthApi.md#finishoauth) | **POST** /oauth/callback |
|
||||||
*OAuthApi* | [**linkOAuthAccount**](doc//OAuthApi.md#linkoauthaccount) | **POST** /oauth/link |
|
*OAuthApi* | [**linkOAuthAccount**](doc//OAuthApi.md#linkoauthaccount) | **POST** /oauth/link |
|
||||||
*OAuthApi* | [**redirectOAuthToMobile**](doc//OAuthApi.md#redirectoauthtomobile) | **GET** /oauth/mobile-redirect |
|
*OAuthApi* | [**redirectOAuthToMobile**](doc//OAuthApi.md#redirectoauthtomobile) | **GET** /oauth/mobile-redirect |
|
||||||
@ -372,6 +373,9 @@ Class | Method | HTTP request | Description
|
|||||||
- [UserDto](doc//UserDto.md)
|
- [UserDto](doc//UserDto.md)
|
||||||
- [UserResponseDto](doc//UserResponseDto.md)
|
- [UserResponseDto](doc//UserResponseDto.md)
|
||||||
- [ValidateAccessTokenResponseDto](doc//ValidateAccessTokenResponseDto.md)
|
- [ValidateAccessTokenResponseDto](doc//ValidateAccessTokenResponseDto.md)
|
||||||
|
- [ValidateLibraryDto](doc//ValidateLibraryDto.md)
|
||||||
|
- [ValidateLibraryImportPathResponseDto](doc//ValidateLibraryImportPathResponseDto.md)
|
||||||
|
- [ValidateLibraryResponseDto](doc//ValidateLibraryResponseDto.md)
|
||||||
- [VideoCodec](doc//VideoCodec.md)
|
- [VideoCodec](doc//VideoCodec.md)
|
||||||
|
|
||||||
|
|
||||||
|
58
mobile/openapi/doc/LibraryApi.md
generated
58
mobile/openapi/doc/LibraryApi.md
generated
@ -17,6 +17,7 @@ Method | HTTP request | Description
|
|||||||
[**removeOfflineFiles**](LibraryApi.md#removeofflinefiles) | **POST** /library/{id}/removeOffline |
|
[**removeOfflineFiles**](LibraryApi.md#removeofflinefiles) | **POST** /library/{id}/removeOffline |
|
||||||
[**scanLibrary**](LibraryApi.md#scanlibrary) | **POST** /library/{id}/scan |
|
[**scanLibrary**](LibraryApi.md#scanlibrary) | **POST** /library/{id}/scan |
|
||||||
[**updateLibrary**](LibraryApi.md#updatelibrary) | **PUT** /library/{id} |
|
[**updateLibrary**](LibraryApi.md#updatelibrary) | **PUT** /library/{id} |
|
||||||
|
[**validate**](LibraryApi.md#validate) | **POST** /library/{id}/validate |
|
||||||
|
|
||||||
|
|
||||||
# **createLibrary**
|
# **createLibrary**
|
||||||
@ -456,3 +457,60 @@ Name | Type | Description | Notes
|
|||||||
|
|
||||||
[[Back to top]](#) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to Model list]](../README.md#documentation-for-models) [[Back to README]](../README.md)
|
[[Back to top]](#) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to Model list]](../README.md#documentation-for-models) [[Back to README]](../README.md)
|
||||||
|
|
||||||
|
# **validate**
|
||||||
|
> ValidateLibraryResponseDto validate(id, validateLibraryDto)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Example
|
||||||
|
```dart
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
// TODO Configure API key authorization: cookie
|
||||||
|
//defaultApiClient.getAuthentication<ApiKeyAuth>('cookie').apiKey = 'YOUR_API_KEY';
|
||||||
|
// uncomment below to setup prefix (e.g. Bearer) for API key, if needed
|
||||||
|
//defaultApiClient.getAuthentication<ApiKeyAuth>('cookie').apiKeyPrefix = 'Bearer';
|
||||||
|
// TODO Configure API key authorization: api_key
|
||||||
|
//defaultApiClient.getAuthentication<ApiKeyAuth>('api_key').apiKey = 'YOUR_API_KEY';
|
||||||
|
// uncomment below to setup prefix (e.g. Bearer) for API key, if needed
|
||||||
|
//defaultApiClient.getAuthentication<ApiKeyAuth>('api_key').apiKeyPrefix = 'Bearer';
|
||||||
|
// TODO Configure HTTP Bearer authorization: bearer
|
||||||
|
// Case 1. Use String Token
|
||||||
|
//defaultApiClient.getAuthentication<HttpBearerAuth>('bearer').setAccessToken('YOUR_ACCESS_TOKEN');
|
||||||
|
// Case 2. Use Function which generate token.
|
||||||
|
// String yourTokenGeneratorFunction() { ... }
|
||||||
|
//defaultApiClient.getAuthentication<HttpBearerAuth>('bearer').setAccessToken(yourTokenGeneratorFunction);
|
||||||
|
|
||||||
|
final api_instance = LibraryApi();
|
||||||
|
final id = 38400000-8cf0-11bd-b23e-10b96e4ef00d; // String |
|
||||||
|
final validateLibraryDto = ValidateLibraryDto(); // ValidateLibraryDto |
|
||||||
|
|
||||||
|
try {
|
||||||
|
final result = api_instance.validate(id, validateLibraryDto);
|
||||||
|
print(result);
|
||||||
|
} catch (e) {
|
||||||
|
print('Exception when calling LibraryApi->validate: $e\n');
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Parameters
|
||||||
|
|
||||||
|
Name | Type | Description | Notes
|
||||||
|
------------- | ------------- | ------------- | -------------
|
||||||
|
**id** | **String**| |
|
||||||
|
**validateLibraryDto** | [**ValidateLibraryDto**](ValidateLibraryDto.md)| |
|
||||||
|
|
||||||
|
### Return type
|
||||||
|
|
||||||
|
[**ValidateLibraryResponseDto**](ValidateLibraryResponseDto.md)
|
||||||
|
|
||||||
|
### Authorization
|
||||||
|
|
||||||
|
[cookie](../README.md#cookie), [api_key](../README.md#api_key), [bearer](../README.md#bearer)
|
||||||
|
|
||||||
|
### HTTP request headers
|
||||||
|
|
||||||
|
- **Content-Type**: application/json
|
||||||
|
- **Accept**: application/json
|
||||||
|
|
||||||
|
[[Back to top]](#) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to Model list]](../README.md#documentation-for-models) [[Back to README]](../README.md)
|
||||||
|
|
||||||
|
16
mobile/openapi/doc/ValidateLibraryDto.md
generated
Normal file
16
mobile/openapi/doc/ValidateLibraryDto.md
generated
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
# openapi.model.ValidateLibraryDto
|
||||||
|
|
||||||
|
## Load the model package
|
||||||
|
```dart
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
```
|
||||||
|
|
||||||
|
## Properties
|
||||||
|
Name | Type | Description | Notes
|
||||||
|
------------ | ------------- | ------------- | -------------
|
||||||
|
**exclusionPatterns** | **List<String>** | | [optional] [default to const []]
|
||||||
|
**importPaths** | **List<String>** | | [optional] [default to const []]
|
||||||
|
|
||||||
|
[[Back to Model list]](../README.md#documentation-for-models) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to README]](../README.md)
|
||||||
|
|
||||||
|
|
17
mobile/openapi/doc/ValidateLibraryImportPathResponseDto.md
generated
Normal file
17
mobile/openapi/doc/ValidateLibraryImportPathResponseDto.md
generated
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# openapi.model.ValidateLibraryImportPathResponseDto
|
||||||
|
|
||||||
|
## Load the model package
|
||||||
|
```dart
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
```
|
||||||
|
|
||||||
|
## Properties
|
||||||
|
Name | Type | Description | Notes
|
||||||
|
------------ | ------------- | ------------- | -------------
|
||||||
|
**importPath** | **String** | |
|
||||||
|
**isValid** | **bool** | | [optional] [default to false]
|
||||||
|
**message** | **String** | | [optional]
|
||||||
|
|
||||||
|
[[Back to Model list]](../README.md#documentation-for-models) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to README]](../README.md)
|
||||||
|
|
||||||
|
|
15
mobile/openapi/doc/ValidateLibraryResponseDto.md
generated
Normal file
15
mobile/openapi/doc/ValidateLibraryResponseDto.md
generated
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# openapi.model.ValidateLibraryResponseDto
|
||||||
|
|
||||||
|
## Load the model package
|
||||||
|
```dart
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
```
|
||||||
|
|
||||||
|
## Properties
|
||||||
|
Name | Type | Description | Notes
|
||||||
|
------------ | ------------- | ------------- | -------------
|
||||||
|
**importPaths** | [**List<ValidateLibraryImportPathResponseDto>**](ValidateLibraryImportPathResponseDto.md) | | [optional] [default to const []]
|
||||||
|
|
||||||
|
[[Back to Model list]](../README.md#documentation-for-models) [[Back to API list]](../README.md#documentation-for-api-endpoints) [[Back to README]](../README.md)
|
||||||
|
|
||||||
|
|
3
mobile/openapi/lib/api.dart
generated
3
mobile/openapi/lib/api.dart
generated
@ -209,6 +209,9 @@ part 'model/user_avatar_color.dart';
|
|||||||
part 'model/user_dto.dart';
|
part 'model/user_dto.dart';
|
||||||
part 'model/user_response_dto.dart';
|
part 'model/user_response_dto.dart';
|
||||||
part 'model/validate_access_token_response_dto.dart';
|
part 'model/validate_access_token_response_dto.dart';
|
||||||
|
part 'model/validate_library_dto.dart';
|
||||||
|
part 'model/validate_library_import_path_response_dto.dart';
|
||||||
|
part 'model/validate_library_response_dto.dart';
|
||||||
part 'model/video_codec.dart';
|
part 'model/video_codec.dart';
|
||||||
|
|
||||||
|
|
||||||
|
52
mobile/openapi/lib/api/library_api.dart
generated
52
mobile/openapi/lib/api/library_api.dart
generated
@ -378,4 +378,56 @@ class LibraryApi {
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Performs an HTTP 'POST /library/{id}/validate' operation and returns the [Response].
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
///
|
||||||
|
/// * [ValidateLibraryDto] validateLibraryDto (required):
|
||||||
|
Future<Response> validateWithHttpInfo(String id, ValidateLibraryDto validateLibraryDto,) async {
|
||||||
|
// ignore: prefer_const_declarations
|
||||||
|
final path = r'/library/{id}/validate'
|
||||||
|
.replaceAll('{id}', id);
|
||||||
|
|
||||||
|
// ignore: prefer_final_locals
|
||||||
|
Object? postBody = validateLibraryDto;
|
||||||
|
|
||||||
|
final queryParams = <QueryParam>[];
|
||||||
|
final headerParams = <String, String>{};
|
||||||
|
final formParams = <String, String>{};
|
||||||
|
|
||||||
|
const contentTypes = <String>['application/json'];
|
||||||
|
|
||||||
|
|
||||||
|
return apiClient.invokeAPI(
|
||||||
|
path,
|
||||||
|
'POST',
|
||||||
|
queryParams,
|
||||||
|
postBody,
|
||||||
|
headerParams,
|
||||||
|
formParams,
|
||||||
|
contentTypes.isEmpty ? null : contentTypes.first,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parameters:
|
||||||
|
///
|
||||||
|
/// * [String] id (required):
|
||||||
|
///
|
||||||
|
/// * [ValidateLibraryDto] validateLibraryDto (required):
|
||||||
|
Future<ValidateLibraryResponseDto?> validate(String id, ValidateLibraryDto validateLibraryDto,) async {
|
||||||
|
final response = await validateWithHttpInfo(id, validateLibraryDto,);
|
||||||
|
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), 'ValidateLibraryResponseDto',) as ValidateLibraryResponseDto;
|
||||||
|
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
6
mobile/openapi/lib/api_client.dart
generated
6
mobile/openapi/lib/api_client.dart
generated
@ -500,6 +500,12 @@ class ApiClient {
|
|||||||
return UserResponseDto.fromJson(value);
|
return UserResponseDto.fromJson(value);
|
||||||
case 'ValidateAccessTokenResponseDto':
|
case 'ValidateAccessTokenResponseDto':
|
||||||
return ValidateAccessTokenResponseDto.fromJson(value);
|
return ValidateAccessTokenResponseDto.fromJson(value);
|
||||||
|
case 'ValidateLibraryDto':
|
||||||
|
return ValidateLibraryDto.fromJson(value);
|
||||||
|
case 'ValidateLibraryImportPathResponseDto':
|
||||||
|
return ValidateLibraryImportPathResponseDto.fromJson(value);
|
||||||
|
case 'ValidateLibraryResponseDto':
|
||||||
|
return ValidateLibraryResponseDto.fromJson(value);
|
||||||
case 'VideoCodec':
|
case 'VideoCodec':
|
||||||
return VideoCodecTypeTransformer().decode(value);
|
return VideoCodecTypeTransformer().decode(value);
|
||||||
default:
|
default:
|
||||||
|
108
mobile/openapi/lib/model/validate_library_dto.dart
generated
Normal file
108
mobile/openapi/lib/model/validate_library_dto.dart
generated
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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 ValidateLibraryDto {
|
||||||
|
/// Returns a new [ValidateLibraryDto] instance.
|
||||||
|
ValidateLibraryDto({
|
||||||
|
this.exclusionPatterns = const [],
|
||||||
|
this.importPaths = const [],
|
||||||
|
});
|
||||||
|
|
||||||
|
List<String> exclusionPatterns;
|
||||||
|
|
||||||
|
List<String> importPaths;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) => identical(this, other) || other is ValidateLibraryDto &&
|
||||||
|
_deepEquality.equals(other.exclusionPatterns, exclusionPatterns) &&
|
||||||
|
_deepEquality.equals(other.importPaths, importPaths);
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
// ignore: unnecessary_parenthesis
|
||||||
|
(exclusionPatterns.hashCode) +
|
||||||
|
(importPaths.hashCode);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'ValidateLibraryDto[exclusionPatterns=$exclusionPatterns, importPaths=$importPaths]';
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final json = <String, dynamic>{};
|
||||||
|
json[r'exclusionPatterns'] = this.exclusionPatterns;
|
||||||
|
json[r'importPaths'] = this.importPaths;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a new [ValidateLibraryDto] instance and imports its values from
|
||||||
|
/// [value] if it's a [Map], null otherwise.
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static ValidateLibraryDto? fromJson(dynamic value) {
|
||||||
|
if (value is Map) {
|
||||||
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
|
return ValidateLibraryDto(
|
||||||
|
exclusionPatterns: json[r'exclusionPatterns'] is Iterable
|
||||||
|
? (json[r'exclusionPatterns'] as Iterable).cast<String>().toList(growable: false)
|
||||||
|
: const [],
|
||||||
|
importPaths: json[r'importPaths'] is Iterable
|
||||||
|
? (json[r'importPaths'] as Iterable).cast<String>().toList(growable: false)
|
||||||
|
: const [],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<ValidateLibraryDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <ValidateLibraryDto>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = ValidateLibraryDto.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Map<String, ValidateLibraryDto> mapFromJson(dynamic json) {
|
||||||
|
final map = <String, ValidateLibraryDto>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
final value = ValidateLibraryDto.fromJson(entry.value);
|
||||||
|
if (value != null) {
|
||||||
|
map[entry.key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
// maps a json object with a list of ValidateLibraryDto-objects as value to a dart map
|
||||||
|
static Map<String, List<ValidateLibraryDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final map = <String, List<ValidateLibraryDto>>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
// ignore: parameter_assignments
|
||||||
|
json = json.cast<String, dynamic>();
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
map[entry.key] = ValidateLibraryDto.listFromJson(entry.value, growable: growable,);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The list of required keys that must be present in a JSON.
|
||||||
|
static const requiredKeys = <String>{
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
122
mobile/openapi/lib/model/validate_library_import_path_response_dto.dart
generated
Normal file
122
mobile/openapi/lib/model/validate_library_import_path_response_dto.dart
generated
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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 ValidateLibraryImportPathResponseDto {
|
||||||
|
/// Returns a new [ValidateLibraryImportPathResponseDto] instance.
|
||||||
|
ValidateLibraryImportPathResponseDto({
|
||||||
|
required this.importPath,
|
||||||
|
this.isValid = false,
|
||||||
|
this.message,
|
||||||
|
});
|
||||||
|
|
||||||
|
String importPath;
|
||||||
|
|
||||||
|
bool isValid;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// Please note: This property should have been non-nullable! Since the specification file
|
||||||
|
/// does not include a default value (using the "default:" property), however, the generated
|
||||||
|
/// source code must fall back to having a nullable type.
|
||||||
|
/// Consider adding a "default:" property in the specification file to hide this note.
|
||||||
|
///
|
||||||
|
String? message;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) => identical(this, other) || other is ValidateLibraryImportPathResponseDto &&
|
||||||
|
other.importPath == importPath &&
|
||||||
|
other.isValid == isValid &&
|
||||||
|
other.message == message;
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
// ignore: unnecessary_parenthesis
|
||||||
|
(importPath.hashCode) +
|
||||||
|
(isValid.hashCode) +
|
||||||
|
(message == null ? 0 : message!.hashCode);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'ValidateLibraryImportPathResponseDto[importPath=$importPath, isValid=$isValid, message=$message]';
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final json = <String, dynamic>{};
|
||||||
|
json[r'importPath'] = this.importPath;
|
||||||
|
json[r'isValid'] = this.isValid;
|
||||||
|
if (this.message != null) {
|
||||||
|
json[r'message'] = this.message;
|
||||||
|
} else {
|
||||||
|
// json[r'message'] = null;
|
||||||
|
}
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a new [ValidateLibraryImportPathResponseDto] instance and imports its values from
|
||||||
|
/// [value] if it's a [Map], null otherwise.
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static ValidateLibraryImportPathResponseDto? fromJson(dynamic value) {
|
||||||
|
if (value is Map) {
|
||||||
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
|
return ValidateLibraryImportPathResponseDto(
|
||||||
|
importPath: mapValueOfType<String>(json, r'importPath')!,
|
||||||
|
isValid: mapValueOfType<bool>(json, r'isValid') ?? false,
|
||||||
|
message: mapValueOfType<String>(json, r'message'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<ValidateLibraryImportPathResponseDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <ValidateLibraryImportPathResponseDto>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = ValidateLibraryImportPathResponseDto.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Map<String, ValidateLibraryImportPathResponseDto> mapFromJson(dynamic json) {
|
||||||
|
final map = <String, ValidateLibraryImportPathResponseDto>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
final value = ValidateLibraryImportPathResponseDto.fromJson(entry.value);
|
||||||
|
if (value != null) {
|
||||||
|
map[entry.key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
// maps a json object with a list of ValidateLibraryImportPathResponseDto-objects as value to a dart map
|
||||||
|
static Map<String, List<ValidateLibraryImportPathResponseDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final map = <String, List<ValidateLibraryImportPathResponseDto>>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
// ignore: parameter_assignments
|
||||||
|
json = json.cast<String, dynamic>();
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
map[entry.key] = ValidateLibraryImportPathResponseDto.listFromJson(entry.value, growable: growable,);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The list of required keys that must be present in a JSON.
|
||||||
|
static const requiredKeys = <String>{
|
||||||
|
'importPath',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
97
mobile/openapi/lib/model/validate_library_response_dto.dart
generated
Normal file
97
mobile/openapi/lib/model/validate_library_response_dto.dart
generated
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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 ValidateLibraryResponseDto {
|
||||||
|
/// Returns a new [ValidateLibraryResponseDto] instance.
|
||||||
|
ValidateLibraryResponseDto({
|
||||||
|
this.importPaths = const [],
|
||||||
|
});
|
||||||
|
|
||||||
|
List<ValidateLibraryImportPathResponseDto> importPaths;
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) => identical(this, other) || other is ValidateLibraryResponseDto &&
|
||||||
|
_deepEquality.equals(other.importPaths, importPaths);
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
// ignore: unnecessary_parenthesis
|
||||||
|
(importPaths.hashCode);
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() => 'ValidateLibraryResponseDto[importPaths=$importPaths]';
|
||||||
|
|
||||||
|
Map<String, dynamic> toJson() {
|
||||||
|
final json = <String, dynamic>{};
|
||||||
|
json[r'importPaths'] = this.importPaths;
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a new [ValidateLibraryResponseDto] instance and imports its values from
|
||||||
|
/// [value] if it's a [Map], null otherwise.
|
||||||
|
// ignore: prefer_constructors_over_static_methods
|
||||||
|
static ValidateLibraryResponseDto? fromJson(dynamic value) {
|
||||||
|
if (value is Map) {
|
||||||
|
final json = value.cast<String, dynamic>();
|
||||||
|
|
||||||
|
return ValidateLibraryResponseDto(
|
||||||
|
importPaths: ValidateLibraryImportPathResponseDto.listFromJson(json[r'importPaths']),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<ValidateLibraryResponseDto> listFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final result = <ValidateLibraryResponseDto>[];
|
||||||
|
if (json is List && json.isNotEmpty) {
|
||||||
|
for (final row in json) {
|
||||||
|
final value = ValidateLibraryResponseDto.fromJson(row);
|
||||||
|
if (value != null) {
|
||||||
|
result.add(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result.toList(growable: growable);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Map<String, ValidateLibraryResponseDto> mapFromJson(dynamic json) {
|
||||||
|
final map = <String, ValidateLibraryResponseDto>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
json = json.cast<String, dynamic>(); // ignore: parameter_assignments
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
final value = ValidateLibraryResponseDto.fromJson(entry.value);
|
||||||
|
if (value != null) {
|
||||||
|
map[entry.key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
// maps a json object with a list of ValidateLibraryResponseDto-objects as value to a dart map
|
||||||
|
static Map<String, List<ValidateLibraryResponseDto>> mapListFromJson(dynamic json, {bool growable = false,}) {
|
||||||
|
final map = <String, List<ValidateLibraryResponseDto>>{};
|
||||||
|
if (json is Map && json.isNotEmpty) {
|
||||||
|
// ignore: parameter_assignments
|
||||||
|
json = json.cast<String, dynamic>();
|
||||||
|
for (final entry in json.entries) {
|
||||||
|
map[entry.key] = ValidateLibraryResponseDto.listFromJson(entry.value, growable: growable,);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return map;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The list of required keys that must be present in a JSON.
|
||||||
|
static const requiredKeys = <String>{
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
5
mobile/openapi/test/library_api_test.dart
generated
5
mobile/openapi/test/library_api_test.dart
generated
@ -57,5 +57,10 @@ void main() {
|
|||||||
// TODO
|
// TODO
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//Future<ValidateLibraryResponseDto> validate(String id, ValidateLibraryDto validateLibraryDto) async
|
||||||
|
test('test validate', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
32
mobile/openapi/test/validate_library_dto_test.dart
generated
Normal file
32
mobile/openapi/test/validate_library_dto_test.dart
generated
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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
|
||||||
|
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
import 'package:test/test.dart';
|
||||||
|
|
||||||
|
// tests for ValidateLibraryDto
|
||||||
|
void main() {
|
||||||
|
// final instance = ValidateLibraryDto();
|
||||||
|
|
||||||
|
group('test ValidateLibraryDto', () {
|
||||||
|
// List<String> exclusionPatterns (default value: const [])
|
||||||
|
test('to test the property `exclusionPatterns`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
// List<String> importPaths (default value: const [])
|
||||||
|
test('to test the property `importPaths`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
37
mobile/openapi/test/validate_library_import_path_response_dto_test.dart
generated
Normal file
37
mobile/openapi/test/validate_library_import_path_response_dto_test.dart
generated
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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
|
||||||
|
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
import 'package:test/test.dart';
|
||||||
|
|
||||||
|
// tests for ValidateLibraryImportPathResponseDto
|
||||||
|
void main() {
|
||||||
|
// final instance = ValidateLibraryImportPathResponseDto();
|
||||||
|
|
||||||
|
group('test ValidateLibraryImportPathResponseDto', () {
|
||||||
|
// String importPath
|
||||||
|
test('to test the property `importPath`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
// bool isValid (default value: false)
|
||||||
|
test('to test the property `isValid`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
// String message
|
||||||
|
test('to test the property `message`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
27
mobile/openapi/test/validate_library_response_dto_test.dart
generated
Normal file
27
mobile/openapi/test/validate_library_response_dto_test.dart
generated
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
//
|
||||||
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
||||||
|
//
|
||||||
|
// @dart=2.12
|
||||||
|
|
||||||
|
// 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
|
||||||
|
|
||||||
|
import 'package:openapi/api.dart';
|
||||||
|
import 'package:test/test.dart';
|
||||||
|
|
||||||
|
// tests for ValidateLibraryResponseDto
|
||||||
|
void main() {
|
||||||
|
// final instance = ValidateLibraryResponseDto();
|
||||||
|
|
||||||
|
group('test ValidateLibraryResponseDto', () {
|
||||||
|
// List<ValidateLibraryImportPathResponseDto> importPaths (default value: const [])
|
||||||
|
test('to test the property `importPaths`', () async {
|
||||||
|
// TODO
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
@ -3618,6 +3618,58 @@
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"/library/{id}/validate": {
|
||||||
|
"post": {
|
||||||
|
"operationId": "validate",
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "id",
|
||||||
|
"required": true,
|
||||||
|
"in": "path",
|
||||||
|
"schema": {
|
||||||
|
"format": "uuid",
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"requestBody": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/ValidateLibraryDto"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": true
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/ValidateLibraryResponseDto"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"description": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"security": [
|
||||||
|
{
|
||||||
|
"bearer": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"cookie": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"api_key": []
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"tags": [
|
||||||
|
"Library"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
"/oauth/authorize": {
|
"/oauth/authorize": {
|
||||||
"post": {
|
"post": {
|
||||||
"operationId": "startOAuth",
|
"operationId": "startOAuth",
|
||||||
@ -10406,6 +10458,52 @@
|
|||||||
],
|
],
|
||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
|
"ValidateLibraryDto": {
|
||||||
|
"properties": {
|
||||||
|
"exclusionPatterns": {
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
},
|
||||||
|
"importPaths": {
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "object"
|
||||||
|
},
|
||||||
|
"ValidateLibraryImportPathResponseDto": {
|
||||||
|
"properties": {
|
||||||
|
"importPath": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
"isValid": {
|
||||||
|
"default": false,
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"message": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"importPath"
|
||||||
|
],
|
||||||
|
"type": "object"
|
||||||
|
},
|
||||||
|
"ValidateLibraryResponseDto": {
|
||||||
|
"properties": {
|
||||||
|
"importPaths": {
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/ValidateLibraryImportPathResponseDto"
|
||||||
|
},
|
||||||
|
"type": "array"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "object"
|
||||||
|
},
|
||||||
"VideoCodec": {
|
"VideoCodec": {
|
||||||
"enum": [
|
"enum": [
|
||||||
"h264",
|
"h264",
|
||||||
|
159
open-api/typescript-sdk/axios-client/api.ts
generated
159
open-api/typescript-sdk/axios-client/api.ts
generated
@ -5316,6 +5316,63 @@ export interface ValidateAccessTokenResponseDto {
|
|||||||
*/
|
*/
|
||||||
'authStatus': boolean;
|
'authStatus': boolean;
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @export
|
||||||
|
* @interface ValidateLibraryDto
|
||||||
|
*/
|
||||||
|
export interface ValidateLibraryDto {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {Array<string>}
|
||||||
|
* @memberof ValidateLibraryDto
|
||||||
|
*/
|
||||||
|
'exclusionPatterns'?: Array<string>;
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {Array<string>}
|
||||||
|
* @memberof ValidateLibraryDto
|
||||||
|
*/
|
||||||
|
'importPaths'?: Array<string>;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @export
|
||||||
|
* @interface ValidateLibraryImportPathResponseDto
|
||||||
|
*/
|
||||||
|
export interface ValidateLibraryImportPathResponseDto {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {string}
|
||||||
|
* @memberof ValidateLibraryImportPathResponseDto
|
||||||
|
*/
|
||||||
|
'importPath': string;
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {boolean}
|
||||||
|
* @memberof ValidateLibraryImportPathResponseDto
|
||||||
|
*/
|
||||||
|
'isValid'?: boolean;
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {string}
|
||||||
|
* @memberof ValidateLibraryImportPathResponseDto
|
||||||
|
*/
|
||||||
|
'message'?: string;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @export
|
||||||
|
* @interface ValidateLibraryResponseDto
|
||||||
|
*/
|
||||||
|
export interface ValidateLibraryResponseDto {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {Array<ValidateLibraryImportPathResponseDto>}
|
||||||
|
* @memberof ValidateLibraryResponseDto
|
||||||
|
*/
|
||||||
|
'importPaths'?: Array<ValidateLibraryImportPathResponseDto>;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @export
|
* @export
|
||||||
@ -12813,6 +12870,54 @@ export const LibraryApiAxiosParamCreator = function (configuration?: Configurati
|
|||||||
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
|
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
|
||||||
localVarRequestOptions.data = serializeDataIfNeeded(updateLibraryDto, localVarRequestOptions, configuration)
|
localVarRequestOptions.data = serializeDataIfNeeded(updateLibraryDto, localVarRequestOptions, configuration)
|
||||||
|
|
||||||
|
return {
|
||||||
|
url: toPathString(localVarUrlObj),
|
||||||
|
options: localVarRequestOptions,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {string} id
|
||||||
|
* @param {ValidateLibraryDto} validateLibraryDto
|
||||||
|
* @param {*} [options] Override http request option.
|
||||||
|
* @throws {RequiredError}
|
||||||
|
*/
|
||||||
|
validate: async (id: string, validateLibraryDto: ValidateLibraryDto, options: RawAxiosRequestConfig = {}): Promise<RequestArgs> => {
|
||||||
|
// verify required parameter 'id' is not null or undefined
|
||||||
|
assertParamExists('validate', 'id', id)
|
||||||
|
// verify required parameter 'validateLibraryDto' is not null or undefined
|
||||||
|
assertParamExists('validate', 'validateLibraryDto', validateLibraryDto)
|
||||||
|
const localVarPath = `/library/{id}/validate`
|
||||||
|
.replace(`{${"id"}}`, encodeURIComponent(String(id)));
|
||||||
|
// use dummy base URL string because the URL constructor only accepts absolute URLs.
|
||||||
|
const localVarUrlObj = new URL(localVarPath, DUMMY_BASE_URL);
|
||||||
|
let baseOptions;
|
||||||
|
if (configuration) {
|
||||||
|
baseOptions = configuration.baseOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
const localVarRequestOptions = { method: 'POST', ...baseOptions, ...options};
|
||||||
|
const localVarHeaderParameter = {} as any;
|
||||||
|
const localVarQueryParameter = {} as any;
|
||||||
|
|
||||||
|
// authentication cookie required
|
||||||
|
|
||||||
|
// authentication api_key required
|
||||||
|
await setApiKeyToObject(localVarHeaderParameter, "x-api-key", configuration)
|
||||||
|
|
||||||
|
// authentication bearer required
|
||||||
|
// http bearer authentication required
|
||||||
|
await setBearerAuthToObject(localVarHeaderParameter, configuration)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
localVarHeaderParameter['Content-Type'] = 'application/json';
|
||||||
|
|
||||||
|
setSearchParams(localVarUrlObj, localVarQueryParameter);
|
||||||
|
let headersFromBaseOptions = baseOptions && baseOptions.headers ? baseOptions.headers : {};
|
||||||
|
localVarRequestOptions.headers = {...localVarHeaderParameter, ...headersFromBaseOptions, ...options.headers};
|
||||||
|
localVarRequestOptions.data = serializeDataIfNeeded(validateLibraryDto, localVarRequestOptions, configuration)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
url: toPathString(localVarUrlObj),
|
url: toPathString(localVarUrlObj),
|
||||||
options: localVarRequestOptions,
|
options: localVarRequestOptions,
|
||||||
@ -12925,6 +13030,19 @@ export const LibraryApiFp = function(configuration?: Configuration) {
|
|||||||
const operationBasePath = operationServerMap['LibraryApi.updateLibrary']?.[index]?.url;
|
const operationBasePath = operationServerMap['LibraryApi.updateLibrary']?.[index]?.url;
|
||||||
return (axios, basePath) => createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration)(axios, operationBasePath || basePath);
|
return (axios, basePath) => createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration)(axios, operationBasePath || basePath);
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {string} id
|
||||||
|
* @param {ValidateLibraryDto} validateLibraryDto
|
||||||
|
* @param {*} [options] Override http request option.
|
||||||
|
* @throws {RequiredError}
|
||||||
|
*/
|
||||||
|
async validate(id: string, validateLibraryDto: ValidateLibraryDto, options?: RawAxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<ValidateLibraryResponseDto>> {
|
||||||
|
const localVarAxiosArgs = await localVarAxiosParamCreator.validate(id, validateLibraryDto, options);
|
||||||
|
const index = configuration?.serverIndex ?? 0;
|
||||||
|
const operationBasePath = operationServerMap['LibraryApi.validate']?.[index]?.url;
|
||||||
|
return (axios, basePath) => createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration)(axios, operationBasePath || basePath);
|
||||||
|
},
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -13006,6 +13124,15 @@ export const LibraryApiFactory = function (configuration?: Configuration, basePa
|
|||||||
updateLibrary(requestParameters: LibraryApiUpdateLibraryRequest, options?: RawAxiosRequestConfig): AxiosPromise<LibraryResponseDto> {
|
updateLibrary(requestParameters: LibraryApiUpdateLibraryRequest, options?: RawAxiosRequestConfig): AxiosPromise<LibraryResponseDto> {
|
||||||
return localVarFp.updateLibrary(requestParameters.id, requestParameters.updateLibraryDto, options).then((request) => request(axios, basePath));
|
return localVarFp.updateLibrary(requestParameters.id, requestParameters.updateLibraryDto, options).then((request) => request(axios, basePath));
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {LibraryApiValidateRequest} requestParameters Request parameters.
|
||||||
|
* @param {*} [options] Override http request option.
|
||||||
|
* @throws {RequiredError}
|
||||||
|
*/
|
||||||
|
validate(requestParameters: LibraryApiValidateRequest, options?: RawAxiosRequestConfig): AxiosPromise<ValidateLibraryResponseDto> {
|
||||||
|
return localVarFp.validate(requestParameters.id, requestParameters.validateLibraryDto, options).then((request) => request(axios, basePath));
|
||||||
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -13121,6 +13248,27 @@ export interface LibraryApiUpdateLibraryRequest {
|
|||||||
readonly updateLibraryDto: UpdateLibraryDto
|
readonly updateLibraryDto: UpdateLibraryDto
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Request parameters for validate operation in LibraryApi.
|
||||||
|
* @export
|
||||||
|
* @interface LibraryApiValidateRequest
|
||||||
|
*/
|
||||||
|
export interface LibraryApiValidateRequest {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {string}
|
||||||
|
* @memberof LibraryApiValidate
|
||||||
|
*/
|
||||||
|
readonly id: string
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @type {ValidateLibraryDto}
|
||||||
|
* @memberof LibraryApiValidate
|
||||||
|
*/
|
||||||
|
readonly validateLibraryDto: ValidateLibraryDto
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* LibraryApi - object-oriented interface
|
* LibraryApi - object-oriented interface
|
||||||
* @export
|
* @export
|
||||||
@ -13214,6 +13362,17 @@ export class LibraryApi extends BaseAPI {
|
|||||||
public updateLibrary(requestParameters: LibraryApiUpdateLibraryRequest, options?: RawAxiosRequestConfig) {
|
public updateLibrary(requestParameters: LibraryApiUpdateLibraryRequest, options?: RawAxiosRequestConfig) {
|
||||||
return LibraryApiFp(this.configuration).updateLibrary(requestParameters.id, requestParameters.updateLibraryDto, options).then((request) => request(this.axios, this.basePath));
|
return LibraryApiFp(this.configuration).updateLibrary(requestParameters.id, requestParameters.updateLibraryDto, options).then((request) => request(this.axios, this.basePath));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {LibraryApiValidateRequest} requestParameters Request parameters.
|
||||||
|
* @param {*} [options] Override http request option.
|
||||||
|
* @throws {RequiredError}
|
||||||
|
* @memberof LibraryApi
|
||||||
|
*/
|
||||||
|
public validate(requestParameters: LibraryApiValidateRequest, options?: RawAxiosRequestConfig) {
|
||||||
|
return LibraryApiFp(this.configuration).validate(requestParameters.id, requestParameters.validateLibraryDto, options).then((request) => request(this.axios, this.basePath));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
25
open-api/typescript-sdk/fetch-client.ts
generated
25
open-api/typescript-sdk/fetch-client.ts
generated
@ -480,6 +480,18 @@ export type LibraryStatsResponseDto = {
|
|||||||
usage: number;
|
usage: number;
|
||||||
videos: number;
|
videos: number;
|
||||||
};
|
};
|
||||||
|
export type ValidateLibraryDto = {
|
||||||
|
exclusionPatterns?: string[];
|
||||||
|
importPaths?: string[];
|
||||||
|
};
|
||||||
|
export type ValidateLibraryImportPathResponseDto = {
|
||||||
|
importPath: string;
|
||||||
|
isValid?: boolean;
|
||||||
|
message?: string;
|
||||||
|
};
|
||||||
|
export type ValidateLibraryResponseDto = {
|
||||||
|
importPaths?: ValidateLibraryImportPathResponseDto[];
|
||||||
|
};
|
||||||
export type OAuthConfigDto = {
|
export type OAuthConfigDto = {
|
||||||
redirectUri: string;
|
redirectUri: string;
|
||||||
};
|
};
|
||||||
@ -1901,6 +1913,19 @@ export function getLibraryStatistics({ id }: {
|
|||||||
...opts
|
...opts
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
export function validate({ id, validateLibraryDto }: {
|
||||||
|
id: string;
|
||||||
|
validateLibraryDto: ValidateLibraryDto;
|
||||||
|
}, opts?: Oazapfts.RequestOpts) {
|
||||||
|
return oazapfts.ok(oazapfts.fetchJson<{
|
||||||
|
status: 200;
|
||||||
|
data: ValidateLibraryResponseDto;
|
||||||
|
}>(`/library/${encodeURIComponent(id)}/validate`, oazapfts.json({
|
||||||
|
...opts,
|
||||||
|
method: "POST",
|
||||||
|
body: validateLibraryDto
|
||||||
|
})));
|
||||||
|
}
|
||||||
export function startOAuth({ oAuthConfigDto }: {
|
export function startOAuth({ oAuthConfigDto }: {
|
||||||
oAuthConfigDto: OAuthConfigDto;
|
oAuthConfigDto: OAuthConfigDto;
|
||||||
}, opts?: Oazapfts.RequestOpts) {
|
}, opts?: Oazapfts.RequestOpts) {
|
||||||
|
@ -1,6 +1,16 @@
|
|||||||
import { PostgreSqlContainer } from '@testcontainers/postgresql';
|
import { PostgreSqlContainer } from '@testcontainers/postgresql';
|
||||||
|
import path from 'node:path';
|
||||||
|
|
||||||
export default async () => {
|
export default async () => {
|
||||||
|
let IMMICH_TEST_ASSET_PATH: string = '';
|
||||||
|
|
||||||
|
if (process.env.IMMICH_TEST_ASSET_PATH === undefined) {
|
||||||
|
IMMICH_TEST_ASSET_PATH = path.normalize(`${__dirname}/../../test/assets/`);
|
||||||
|
process.env.IMMICH_TEST_ASSET_PATH = IMMICH_TEST_ASSET_PATH;
|
||||||
|
} else {
|
||||||
|
IMMICH_TEST_ASSET_PATH = process.env.IMMICH_TEST_ASSET_PATH;
|
||||||
|
}
|
||||||
|
|
||||||
const pg = await new PostgreSqlContainer('tensorchord/pgvecto-rs:pg14-v0.2.0')
|
const pg = await new PostgreSqlContainer('tensorchord/pgvecto-rs:pg14-v0.2.0')
|
||||||
.withDatabase('immich')
|
.withDatabase('immich')
|
||||||
.withUsername('postgres')
|
.withUsername('postgres')
|
||||||
@ -11,6 +21,9 @@ export default async () => {
|
|||||||
|
|
||||||
process.env.DB_URL = pg.getConnectionUri();
|
process.env.DB_URL = pg.getConnectionUri();
|
||||||
process.env.NODE_ENV = 'development';
|
process.env.NODE_ENV = 'development';
|
||||||
process.env.LOG_LEVEL = 'fatal';
|
|
||||||
process.env.TZ = 'Z';
|
process.env.TZ = 'Z';
|
||||||
|
|
||||||
|
if (process.env.LOG_LEVEL === undefined) {
|
||||||
|
process.env.LOG_LEVEL = 'fatal';
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
@ -2,6 +2,7 @@ import { LibraryResponseDto, LoginResponseDto } from '@app/domain';
|
|||||||
import { LibraryController } from '@app/immich';
|
import { LibraryController } from '@app/immich';
|
||||||
import { LibraryType } from '@app/infra/entities';
|
import { LibraryType } from '@app/infra/entities';
|
||||||
import { errorStub, userDto, uuidStub } from '@test/fixtures';
|
import { errorStub, userDto, uuidStub } from '@test/fixtures';
|
||||||
|
import { IMMICH_TEST_ASSET_TEMP_PATH, restoreTempFolder } from 'src/test-utils/utils';
|
||||||
import request from 'supertest';
|
import request from 'supertest';
|
||||||
import { api } from '../../client';
|
import { api } from '../../client';
|
||||||
import { testApp } from '../utils';
|
import { testApp } from '../utils';
|
||||||
@ -20,6 +21,7 @@ describe(`${LibraryController.name} (e2e)`, () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
|
await restoreTempFolder();
|
||||||
await testApp.reset();
|
await testApp.reset();
|
||||||
await api.authApi.adminSignUp(server);
|
await api.authApi.adminSignUp(server);
|
||||||
admin = await api.authApi.adminLogin(server);
|
admin = await api.authApi.adminLogin(server);
|
||||||
@ -247,15 +249,16 @@ describe(`${LibraryController.name} (e2e)`, () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should change the import paths', async () => {
|
it('should change the import paths', async () => {
|
||||||
|
await api.userApi.setExternalPath(server, admin.accessToken, admin.userId, IMMICH_TEST_ASSET_TEMP_PATH);
|
||||||
const { status, body } = await request(server)
|
const { status, body } = await request(server)
|
||||||
.put(`/library/${library.id}`)
|
.put(`/library/${library.id}`)
|
||||||
.set('Authorization', `Bearer ${admin.accessToken}`)
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
||||||
.send({ importPaths: ['/path/to/import'] });
|
.send({ importPaths: [IMMICH_TEST_ASSET_TEMP_PATH] });
|
||||||
|
|
||||||
expect(status).toBe(200);
|
expect(status).toBe(200);
|
||||||
expect(body).toEqual(
|
expect(body).toEqual(
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
importPaths: ['/path/to/import'],
|
importPaths: [IMMICH_TEST_ASSET_TEMP_PATH],
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
@ -435,4 +438,93 @@ describe(`${LibraryController.name} (e2e)`, () => {
|
|||||||
expect(body).toEqual(errorStub.unauthorized);
|
expect(body).toEqual(errorStub.unauthorized);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('POST /library/:id/validate', () => {
|
||||||
|
it('should require authentication', async () => {
|
||||||
|
const { status, body } = await request(server).post(`/library/${uuidStub.notFound}/validate`).send({});
|
||||||
|
|
||||||
|
expect(status).toBe(401);
|
||||||
|
expect(body).toEqual(errorStub.unauthorized);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Validate import path', () => {
|
||||||
|
let library: LibraryResponseDto;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
// Create an external library with default settings
|
||||||
|
library = await api.libraryApi.create(server, admin.accessToken, { type: LibraryType.EXTERNAL });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should fail with no external path set', async () => {
|
||||||
|
const { status, body } = await request(server)
|
||||||
|
.post(`/library/${library.id}/validate`)
|
||||||
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
||||||
|
|
||||||
|
.send({ importPaths: [] });
|
||||||
|
|
||||||
|
expect(status).toBe(400);
|
||||||
|
expect(body).toEqual(errorStub.badRequest('User has no external path set'));
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('With external path set', () => {
|
||||||
|
beforeEach(async () => {
|
||||||
|
await api.userApi.setExternalPath(server, admin.accessToken, admin.userId, IMMICH_TEST_ASSET_TEMP_PATH);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should pass with no import paths', async () => {
|
||||||
|
const response = await api.libraryApi.validate(server, admin.accessToken, library.id, { importPaths: [] });
|
||||||
|
expect(response.importPaths).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not allow paths outside of the external path', async () => {
|
||||||
|
const pathToTest = `${IMMICH_TEST_ASSET_TEMP_PATH}/../`;
|
||||||
|
const response = await api.libraryApi.validate(server, admin.accessToken, library.id, {
|
||||||
|
importPaths: [pathToTest],
|
||||||
|
});
|
||||||
|
expect(response.importPaths?.length).toEqual(1);
|
||||||
|
const pathResponse = response?.importPaths?.at(0);
|
||||||
|
|
||||||
|
expect(pathResponse).toEqual({
|
||||||
|
importPath: pathToTest,
|
||||||
|
isValid: false,
|
||||||
|
message: `Not contained in user's external path`,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should fail if path does not exist', async () => {
|
||||||
|
const pathToTest = `${IMMICH_TEST_ASSET_TEMP_PATH}/does/not/exist`;
|
||||||
|
|
||||||
|
const response = await api.libraryApi.validate(server, admin.accessToken, library.id, {
|
||||||
|
importPaths: [pathToTest],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.importPaths?.length).toEqual(1);
|
||||||
|
const pathResponse = response?.importPaths?.at(0);
|
||||||
|
|
||||||
|
expect(pathResponse).toEqual({
|
||||||
|
importPath: pathToTest,
|
||||||
|
isValid: false,
|
||||||
|
message: `Path does not exist (ENOENT)`,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should fail if path is a file', async () => {
|
||||||
|
const pathToTest = `${IMMICH_TEST_ASSET_TEMP_PATH}/does/not/exist`;
|
||||||
|
|
||||||
|
const response = await api.libraryApi.validate(server, admin.accessToken, library.id, {
|
||||||
|
importPaths: [pathToTest],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.importPaths?.length).toEqual(1);
|
||||||
|
const pathResponse = response?.importPaths?.at(0);
|
||||||
|
|
||||||
|
expect(pathResponse).toEqual({
|
||||||
|
importPath: pathToTest,
|
||||||
|
isValid: false,
|
||||||
|
message: `Path does not exist (ENOENT)`,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -4,6 +4,8 @@ import {
|
|||||||
LibraryStatsResponseDto,
|
LibraryStatsResponseDto,
|
||||||
ScanLibraryDto,
|
ScanLibraryDto,
|
||||||
UpdateLibraryDto,
|
UpdateLibraryDto,
|
||||||
|
ValidateLibraryDto,
|
||||||
|
ValidateLibraryResponseDto,
|
||||||
} from '@app/domain';
|
} from '@app/domain';
|
||||||
import request from 'supertest';
|
import request from 'supertest';
|
||||||
|
|
||||||
@ -58,4 +60,12 @@ export const libraryApi = {
|
|||||||
expect(status).toBe(200);
|
expect(status).toBe(200);
|
||||||
return body as LibraryResponseDto;
|
return body as LibraryResponseDto;
|
||||||
},
|
},
|
||||||
|
validate: async (server: any, accessToken: string, id: string, data: ValidateLibraryDto) => {
|
||||||
|
const { body, status } = await request(server)
|
||||||
|
.post(`/library/${id}/validate`)
|
||||||
|
.set('Authorization', `Bearer ${accessToken}`)
|
||||||
|
.send(data);
|
||||||
|
expect(status).toBe(200);
|
||||||
|
return body as ValidateLibraryResponseDto;
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { LibraryEntity, LibraryType } from '@app/infra/entities';
|
import { LibraryEntity, LibraryType } from '@app/infra/entities';
|
||||||
import { ApiProperty } from '@nestjs/swagger';
|
import { ApiProperty } from '@nestjs/swagger';
|
||||||
import { ArrayUnique, IsBoolean, IsEnum, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
import { ArrayMaxSize, ArrayUnique, IsBoolean, IsEnum, IsNotEmpty, IsOptional, IsString } from 'class-validator';
|
||||||
import { ValidateUUID } from '../domain.util';
|
import { ValidateUUID } from '../domain.util';
|
||||||
|
|
||||||
export class CreateLibraryDto {
|
export class CreateLibraryDto {
|
||||||
@ -21,12 +21,14 @@ export class CreateLibraryDto {
|
|||||||
@IsString({ each: true })
|
@IsString({ each: true })
|
||||||
@IsNotEmpty({ each: true })
|
@IsNotEmpty({ each: true })
|
||||||
@ArrayUnique()
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
importPaths?: string[];
|
importPaths?: string[];
|
||||||
|
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsString({ each: true })
|
@IsString({ each: true })
|
||||||
@IsNotEmpty({ each: true })
|
@IsNotEmpty({ each: true })
|
||||||
@ArrayUnique()
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
exclusionPatterns?: string[];
|
exclusionPatterns?: string[];
|
||||||
|
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@ -48,12 +50,14 @@ export class UpdateLibraryDto {
|
|||||||
@IsString({ each: true })
|
@IsString({ each: true })
|
||||||
@IsNotEmpty({ each: true })
|
@IsNotEmpty({ each: true })
|
||||||
@ArrayUnique()
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
importPaths?: string[];
|
importPaths?: string[];
|
||||||
|
|
||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsNotEmpty({ each: true })
|
@IsNotEmpty({ each: true })
|
||||||
@IsString({ each: true })
|
@IsString({ each: true })
|
||||||
@ArrayUnique()
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
exclusionPatterns?: string[];
|
exclusionPatterns?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -63,6 +67,32 @@ export class CrawlOptionsDto {
|
|||||||
exclusionPatterns?: string[];
|
exclusionPatterns?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class ValidateLibraryDto {
|
||||||
|
@IsOptional()
|
||||||
|
@IsString({ each: true })
|
||||||
|
@IsNotEmpty({ each: true })
|
||||||
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
|
importPaths?: string[];
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsNotEmpty({ each: true })
|
||||||
|
@IsString({ each: true })
|
||||||
|
@ArrayUnique()
|
||||||
|
@ArrayMaxSize(128)
|
||||||
|
exclusionPatterns?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ValidateLibraryResponseDto {
|
||||||
|
importPaths?: ValidateLibraryImportPathResponseDto[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ValidateLibraryImportPathResponseDto {
|
||||||
|
importPath!: string;
|
||||||
|
isValid?: boolean = false;
|
||||||
|
message?: string;
|
||||||
|
}
|
||||||
|
|
||||||
export class LibrarySearchDto {
|
export class LibrarySearchDto {
|
||||||
@ValidateUUID({ optional: true })
|
@ValidateUUID({ optional: true })
|
||||||
userId?: string;
|
userId?: string;
|
||||||
|
@ -54,12 +54,6 @@ describe(LibraryService.name, () => {
|
|||||||
cryptoMock = newCryptoRepositoryMock();
|
cryptoMock = newCryptoRepositoryMock();
|
||||||
storageMock = newStorageRepositoryMock();
|
storageMock = newStorageRepositoryMock();
|
||||||
|
|
||||||
storageMock.stat.mockResolvedValue({
|
|
||||||
size: 100,
|
|
||||||
mtime: new Date('2023-01-01'),
|
|
||||||
ctime: new Date('2023-01-01'),
|
|
||||||
} as Stats);
|
|
||||||
|
|
||||||
// Always validate owner access for library.
|
// Always validate owner access for library.
|
||||||
accessMock.library.checkOwnerAccess.mockImplementation(async (_, libraryIds) => libraryIds);
|
accessMock.library.checkOwnerAccess.mockImplementation(async (_, libraryIds) => libraryIds);
|
||||||
|
|
||||||
@ -270,6 +264,39 @@ describe(LibraryService.name, () => {
|
|||||||
|
|
||||||
await expect(sut.handleQueueAssetRefresh(mockLibraryJob)).resolves.toBe(false);
|
await expect(sut.handleQueueAssetRefresh(mockLibraryJob)).resolves.toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should ignore import paths that do not exist', async () => {
|
||||||
|
storageMock.stat.mockImplementation((path): Promise<Stats> => {
|
||||||
|
if (path === libraryStub.externalLibraryWithImportPaths1.importPaths[0]) {
|
||||||
|
const error = { code: 'ENOENT' } as any;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
return Promise.resolve({
|
||||||
|
isDirectory: () => true,
|
||||||
|
} as Stats);
|
||||||
|
});
|
||||||
|
|
||||||
|
storageMock.checkFileExists.mockResolvedValue(true);
|
||||||
|
|
||||||
|
const mockLibraryJob: ILibraryRefreshJob = {
|
||||||
|
id: libraryStub.externalLibraryWithImportPaths1.id,
|
||||||
|
refreshModifiedFiles: false,
|
||||||
|
refreshAllFiles: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
libraryMock.get.mockResolvedValue(libraryStub.externalLibraryWithImportPaths1);
|
||||||
|
storageMock.crawl.mockResolvedValue([]);
|
||||||
|
assetMock.getByLibraryId.mockResolvedValue([]);
|
||||||
|
libraryMock.getOnlineAssetPaths.mockResolvedValue([]);
|
||||||
|
userMock.get.mockResolvedValue(userStub.externalPathRoot);
|
||||||
|
|
||||||
|
await sut.handleQueueAssetRefresh(mockLibraryJob);
|
||||||
|
|
||||||
|
expect(storageMock.crawl).toHaveBeenCalledWith({
|
||||||
|
pathsToCrawl: [libraryStub.externalLibraryWithImportPaths1.importPaths[1]],
|
||||||
|
exclusionPatterns: [],
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('handleAssetRefresh', () => {
|
describe('handleAssetRefresh', () => {
|
||||||
@ -278,6 +305,12 @@ describe(LibraryService.name, () => {
|
|||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mockUser = userStub.externalPath1;
|
mockUser = userStub.externalPath1;
|
||||||
userMock.get.mockResolvedValue(mockUser);
|
userMock.get.mockResolvedValue(mockUser);
|
||||||
|
|
||||||
|
storageMock.stat.mockResolvedValue({
|
||||||
|
size: 100,
|
||||||
|
mtime: new Date('2023-01-01'),
|
||||||
|
ctime: new Date('2023-01-01'),
|
||||||
|
} as Stats);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should reject an unknown file extension', async () => {
|
it('should reject an unknown file extension', async () => {
|
||||||
@ -1104,13 +1137,19 @@ describe(LibraryService.name, () => {
|
|||||||
libraryMock.update.mockResolvedValue(libraryStub.externalLibraryWithImportPaths1);
|
libraryMock.update.mockResolvedValue(libraryStub.externalLibraryWithImportPaths1);
|
||||||
libraryMock.get.mockResolvedValue(libraryStub.externalLibraryWithImportPaths1);
|
libraryMock.get.mockResolvedValue(libraryStub.externalLibraryWithImportPaths1);
|
||||||
|
|
||||||
await expect(sut.update(authStub.admin, authStub.admin.user.id, { importPaths: ['/foo'] })).resolves.toEqual(
|
storageMock.stat.mockResolvedValue({
|
||||||
mapLibrary(libraryStub.externalLibraryWithImportPaths1),
|
isDirectory: () => true,
|
||||||
);
|
} as Stats);
|
||||||
|
|
||||||
|
storageMock.checkFileExists.mockResolvedValue(true);
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
sut.update(authStub.external1, authStub.external1.user.id, { importPaths: ['/data/user1/foo'] }),
|
||||||
|
).resolves.toEqual(mapLibrary(libraryStub.externalLibraryWithImportPaths1));
|
||||||
|
|
||||||
expect(libraryMock.update).toHaveBeenCalledWith(
|
expect(libraryMock.update).toHaveBeenCalledWith(
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
id: authStub.admin.user.id,
|
id: authStub.external1.user.id,
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
expect(storageMock.watch).toHaveBeenCalledWith(
|
expect(storageMock.watch).toHaveBeenCalledWith(
|
||||||
@ -1142,7 +1181,7 @@ describe(LibraryService.name, () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('watchAll new', () => {
|
describe('watchAll', () => {
|
||||||
describe('watching disabled', () => {
|
describe('watching disabled', () => {
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
configMock.load.mockResolvedValue(systemConfigStub.libraryWatchDisabled);
|
configMock.load.mockResolvedValue(systemConfigStub.libraryWatchDisabled);
|
||||||
@ -1523,4 +1562,121 @@ describe(LibraryService.name, () => {
|
|||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('validate', () => {
|
||||||
|
it('should validate directory', async () => {
|
||||||
|
storageMock.stat.mockResolvedValue({
|
||||||
|
isDirectory: () => true,
|
||||||
|
} as Stats);
|
||||||
|
|
||||||
|
storageMock.checkFileExists.mockResolvedValue(true);
|
||||||
|
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user1/'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user1/',
|
||||||
|
isValid: true,
|
||||||
|
message: undefined,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should error when no external path is set', async () => {
|
||||||
|
await expect(
|
||||||
|
sut.validate(authStub.admin, libraryStub.externalLibrary1.id, { importPaths: ['/photos'] }),
|
||||||
|
).rejects.toBeInstanceOf(BadRequestException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should detect when path is outside external path', async () => {
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user2'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user2',
|
||||||
|
isValid: false,
|
||||||
|
message: "Not contained in user's external path",
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should detect when path does not exist', async () => {
|
||||||
|
storageMock.stat.mockImplementation(() => {
|
||||||
|
const error = { code: 'ENOENT' } as any;
|
||||||
|
throw error;
|
||||||
|
});
|
||||||
|
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user1/'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user1/',
|
||||||
|
isValid: false,
|
||||||
|
message: 'Path does not exist (ENOENT)',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should detect when path is not a directory', async () => {
|
||||||
|
storageMock.stat.mockResolvedValue({
|
||||||
|
isDirectory: () => false,
|
||||||
|
} as Stats);
|
||||||
|
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user1/file'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user1/file',
|
||||||
|
isValid: false,
|
||||||
|
message: 'Not a directory',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return an unknown exception from stat', async () => {
|
||||||
|
storageMock.stat.mockImplementation(() => {
|
||||||
|
throw new Error('Unknown error');
|
||||||
|
});
|
||||||
|
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user1/'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user1/',
|
||||||
|
isValid: false,
|
||||||
|
message: 'Error: Unknown error',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should detect when access rights are missing', async () => {
|
||||||
|
storageMock.stat.mockResolvedValue({
|
||||||
|
isDirectory: () => true,
|
||||||
|
} as Stats);
|
||||||
|
|
||||||
|
storageMock.checkFileExists.mockResolvedValue(false);
|
||||||
|
|
||||||
|
const result = await sut.validate(authStub.external1, libraryStub.externalLibraryWithImportPaths1.id, {
|
||||||
|
importPaths: ['/data/user1/'],
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(result.importPaths).toEqual([
|
||||||
|
{
|
||||||
|
importPath: '/data/user1/',
|
||||||
|
isValid: false,
|
||||||
|
message: 'Lacking read permission for folder',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -30,6 +30,9 @@ import {
|
|||||||
LibraryStatsResponseDto,
|
LibraryStatsResponseDto,
|
||||||
ScanLibraryDto,
|
ScanLibraryDto,
|
||||||
UpdateLibraryDto,
|
UpdateLibraryDto,
|
||||||
|
ValidateLibraryDto,
|
||||||
|
ValidateLibraryImportPathResponseDto,
|
||||||
|
ValidateLibraryResponseDto,
|
||||||
mapLibrary,
|
mapLibrary,
|
||||||
} from './library.dto';
|
} from './library.dto';
|
||||||
|
|
||||||
@ -270,10 +273,81 @@ export class LibraryService extends EventEmitter {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async validateImportPath(importPath: string): Promise<ValidateLibraryImportPathResponseDto> {
|
||||||
|
const validation = new ValidateLibraryImportPathResponseDto();
|
||||||
|
validation.importPath = importPath;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const stat = await this.storageRepository.stat(importPath);
|
||||||
|
|
||||||
|
if (!stat.isDirectory()) {
|
||||||
|
validation.message = 'Not a directory';
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
if (error.code === 'ENOENT') {
|
||||||
|
validation.message = 'Path does not exist (ENOENT)';
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
validation.message = String(error);
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
|
||||||
|
const access = await this.storageRepository.checkFileExists(importPath, R_OK);
|
||||||
|
|
||||||
|
if (!access) {
|
||||||
|
validation.message = 'Lacking read permission for folder';
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
|
||||||
|
validation.isValid = true;
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async validate(auth: AuthDto, id: string, dto: ValidateLibraryDto): Promise<ValidateLibraryResponseDto> {
|
||||||
|
await this.access.requirePermission(auth, Permission.LIBRARY_UPDATE, id);
|
||||||
|
|
||||||
|
if (!auth.user.externalPath) {
|
||||||
|
throw new BadRequestException('User has no external path set');
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = new ValidateLibraryResponseDto();
|
||||||
|
|
||||||
|
if (dto.importPaths) {
|
||||||
|
response.importPaths = await Promise.all(
|
||||||
|
dto.importPaths.map(async (importPath) => {
|
||||||
|
const normalizedPath = path.normalize(importPath);
|
||||||
|
|
||||||
|
if (!this.isInExternalPath(normalizedPath, auth.user.externalPath)) {
|
||||||
|
const validation = new ValidateLibraryImportPathResponseDto();
|
||||||
|
validation.importPath = importPath;
|
||||||
|
validation.message = `Not contained in user's external path`;
|
||||||
|
return validation;
|
||||||
|
}
|
||||||
|
|
||||||
|
return await this.validateImportPath(importPath);
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
async update(auth: AuthDto, id: string, dto: UpdateLibraryDto): Promise<LibraryResponseDto> {
|
async update(auth: AuthDto, id: string, dto: UpdateLibraryDto): Promise<LibraryResponseDto> {
|
||||||
await this.access.requirePermission(auth, Permission.LIBRARY_UPDATE, id);
|
await this.access.requirePermission(auth, Permission.LIBRARY_UPDATE, id);
|
||||||
const library = await this.repository.update({ id, ...dto });
|
const library = await this.repository.update({ id, ...dto });
|
||||||
|
|
||||||
|
if (dto.importPaths) {
|
||||||
|
const validation = await this.validate(auth, id, { importPaths: dto.importPaths });
|
||||||
|
if (validation.importPaths) {
|
||||||
|
for (const path of validation.importPaths) {
|
||||||
|
if (!path.isValid) {
|
||||||
|
throw new BadRequestException(`Invalid import path: ${path.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (dto.importPaths || dto.exclusionPatterns) {
|
if (dto.importPaths || dto.exclusionPatterns) {
|
||||||
// Re-watch library to use new paths and/or exclusion patterns
|
// Re-watch library to use new paths and/or exclusion patterns
|
||||||
await this.watch(id);
|
await this.watch(id);
|
||||||
@ -509,6 +583,14 @@ export class LibraryService extends EventEmitter {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check if a given path is in a user's external path. Both arguments are assumed to be normalized
|
||||||
|
private isInExternalPath(filePath: string, externalPath: string | null): boolean {
|
||||||
|
if (externalPath === null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return filePath.startsWith(externalPath);
|
||||||
|
}
|
||||||
|
|
||||||
async handleQueueAssetRefresh(job: ILibraryRefreshJob): Promise<boolean> {
|
async handleQueueAssetRefresh(job: ILibraryRefreshJob): Promise<boolean> {
|
||||||
const library = await this.repository.get(job.id);
|
const library = await this.repository.get(job.id);
|
||||||
if (!library || library.type !== LibraryType.EXTERNAL) {
|
if (!library || library.type !== LibraryType.EXTERNAL) {
|
||||||
@ -523,17 +605,31 @@ export class LibraryService extends EventEmitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.logger.verbose(`Refreshing library: ${job.id}`);
|
this.logger.verbose(`Refreshing library: ${job.id}`);
|
||||||
|
|
||||||
|
const pathValidation = await Promise.all(
|
||||||
|
library.importPaths.map(async (importPath) => await this.validateImportPath(importPath)),
|
||||||
|
);
|
||||||
|
|
||||||
|
const validImportPaths = pathValidation
|
||||||
|
.map((validation) => {
|
||||||
|
if (!validation.isValid) {
|
||||||
|
this.logger.error(`Skipping invalid import path: ${validation.importPath}. Reason: ${validation.message}`);
|
||||||
|
}
|
||||||
|
return validation;
|
||||||
|
})
|
||||||
|
.filter((validation) => validation.isValid)
|
||||||
|
.map((validation) => validation.importPath);
|
||||||
|
|
||||||
const rawPaths = await this.storageRepository.crawl({
|
const rawPaths = await this.storageRepository.crawl({
|
||||||
pathsToCrawl: library.importPaths,
|
pathsToCrawl: validImportPaths,
|
||||||
exclusionPatterns: library.exclusionPatterns,
|
exclusionPatterns: library.exclusionPatterns,
|
||||||
});
|
});
|
||||||
|
|
||||||
const crawledAssetPaths = rawPaths
|
const crawledAssetPaths = rawPaths
|
||||||
|
// Normalize file paths. This is important to prevent security issues like path traversal
|
||||||
.map((filePath) => path.normalize(filePath))
|
.map((filePath) => path.normalize(filePath))
|
||||||
.filter((assetPath) =>
|
// Filter out paths that are not within the user's external path
|
||||||
// Filter out paths that are not within the user's external path
|
.filter((assetPath) => this.isInExternalPath(assetPath, user.externalPath)) as string[];
|
||||||
assetPath.match(new RegExp(`^${user.externalPath}`)),
|
|
||||||
) as string[];
|
|
||||||
|
|
||||||
this.logger.debug(`Found ${crawledAssetPaths.length} asset(s) when crawling import paths ${library.importPaths}`);
|
this.logger.debug(`Found ${crawledAssetPaths.length} asset(s) when crawling import paths ${library.importPaths}`);
|
||||||
const assetsInLibrary = await this.assetRepository.getByLibraryId([job.id]);
|
const assetsInLibrary = await this.assetRepository.getByLibraryId([job.id]);
|
||||||
|
@ -6,8 +6,10 @@ import {
|
|||||||
LibraryResponseDto as ResponseDto,
|
LibraryResponseDto as ResponseDto,
|
||||||
ScanLibraryDto,
|
ScanLibraryDto,
|
||||||
UpdateLibraryDto as UpdateDto,
|
UpdateLibraryDto as UpdateDto,
|
||||||
|
ValidateLibraryDto,
|
||||||
|
ValidateLibraryResponseDto,
|
||||||
} from '@app/domain';
|
} from '@app/domain';
|
||||||
import { Body, Controller, Delete, Get, Param, Post, Put } from '@nestjs/common';
|
import { Body, Controller, Delete, Get, HttpCode, Param, Post, Put } from '@nestjs/common';
|
||||||
import { ApiTags } from '@nestjs/swagger';
|
import { ApiTags } from '@nestjs/swagger';
|
||||||
import { Auth, Authenticated } from '../app.guard';
|
import { Auth, Authenticated } from '../app.guard';
|
||||||
import { UseValidation } from '../app.utils';
|
import { UseValidation } from '../app.utils';
|
||||||
@ -40,6 +42,16 @@ export class LibraryController {
|
|||||||
return this.service.get(auth, id);
|
return this.service.get(auth, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Post(':id/validate')
|
||||||
|
@HttpCode(200)
|
||||||
|
validate(
|
||||||
|
@Auth() auth: AuthDto,
|
||||||
|
@Param() { id }: UUIDParamDto,
|
||||||
|
@Body() dto: ValidateLibraryDto,
|
||||||
|
): Promise<ValidateLibraryResponseDto> {
|
||||||
|
return this.service.validate(auth, id, dto);
|
||||||
|
}
|
||||||
|
|
||||||
@Delete(':id')
|
@Delete(':id')
|
||||||
deleteLibrary(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto): Promise<void> {
|
deleteLibrary(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto): Promise<void> {
|
||||||
return this.service.delete(auth, id);
|
return this.service.delete(auth, id);
|
||||||
|
@ -164,7 +164,21 @@ export function searchAssetBuilder(
|
|||||||
builder.andWhere(_.omitBy(path, _.isUndefined));
|
builder.andWhere(_.omitBy(path, _.isUndefined));
|
||||||
|
|
||||||
const status = _.pick(options, ['isExternal', 'isFavorite', 'isOffline', 'isReadOnly', 'isVisible', 'type']);
|
const status = _.pick(options, ['isExternal', 'isFavorite', 'isOffline', 'isReadOnly', 'isVisible', 'type']);
|
||||||
const { isArchived, isEncoded, isMotion, withArchived } = options;
|
const {
|
||||||
|
isArchived,
|
||||||
|
isEncoded,
|
||||||
|
isMotion,
|
||||||
|
withArchived,
|
||||||
|
isNotInAlbum,
|
||||||
|
withFaces,
|
||||||
|
withPeople,
|
||||||
|
withSmartInfo,
|
||||||
|
personIds,
|
||||||
|
withExif,
|
||||||
|
withStacked,
|
||||||
|
trashedAfter,
|
||||||
|
trashedBefore,
|
||||||
|
} = options;
|
||||||
builder.andWhere(
|
builder.andWhere(
|
||||||
_.omitBy(
|
_.omitBy(
|
||||||
{
|
{
|
||||||
@ -177,38 +191,38 @@ export function searchAssetBuilder(
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (options.isNotInAlbum) {
|
if (isNotInAlbum) {
|
||||||
builder
|
builder
|
||||||
.leftJoin(`${builder.alias}.albums`, 'albums')
|
.leftJoin(`${builder.alias}.albums`, 'albums')
|
||||||
.andWhere('albums.id IS NULL')
|
.andWhere('albums.id IS NULL')
|
||||||
.andWhere(`${builder.alias}.isVisible = true`);
|
.andWhere(`${builder.alias}.isVisible = true`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.withFaces || options.withPeople) {
|
if (withFaces || withPeople) {
|
||||||
builder.leftJoinAndSelect(`${builder.alias}.faces`, 'faces');
|
builder.leftJoinAndSelect(`${builder.alias}.faces`, 'faces');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.withPeople) {
|
if (withPeople) {
|
||||||
builder.leftJoinAndSelect(`${builder.alias}.person`, 'person');
|
builder.leftJoinAndSelect(`${builder.alias}.person`, 'person');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.withSmartInfo) {
|
if (withSmartInfo) {
|
||||||
builder.leftJoinAndSelect(`${builder.alias}.smartInfo`, 'smartInfo');
|
builder.leftJoinAndSelect(`${builder.alias}.smartInfo`, 'smartInfo');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.personIds && options.personIds.length > 0) {
|
if (personIds && personIds.length > 0) {
|
||||||
builder
|
builder
|
||||||
.leftJoin(`${builder.alias}.faces`, 'faces')
|
.leftJoin(`${builder.alias}.faces`, 'faces')
|
||||||
.andWhere('faces.personId IN (:...personIds)', { personIds: options.personIds })
|
.andWhere('faces.personId IN (:...personIds)', { personIds: personIds })
|
||||||
.addGroupBy(`${builder.alias}.id`)
|
.addGroupBy(`${builder.alias}.id`)
|
||||||
.having('COUNT(faces.id) = :personCount', { personCount: options.personIds.length });
|
.having('COUNT(faces.id) = :personCount', { personCount: personIds.length });
|
||||||
|
|
||||||
if (options.withExif) {
|
if (withExif) {
|
||||||
builder.addGroupBy('exifInfo.assetId');
|
builder.addGroupBy('exifInfo.assetId');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options.withStacked) {
|
if (withStacked) {
|
||||||
builder
|
builder
|
||||||
.leftJoinAndSelect(`${builder.alias}.stack`, 'stack')
|
.leftJoinAndSelect(`${builder.alias}.stack`, 'stack')
|
||||||
.leftJoinAndSelect('stack.assets', 'stackedAssets')
|
.leftJoinAndSelect('stack.assets', 'stackedAssets')
|
||||||
@ -217,8 +231,7 @@ export function searchAssetBuilder(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const withDeleted =
|
const withDeleted = options.withDeleted ?? (trashedAfter !== undefined || trashedBefore !== undefined);
|
||||||
options.withDeleted ?? (options.trashedAfter !== undefined || options.trashedBefore !== undefined);
|
|
||||||
if (withDeleted) {
|
if (withDeleted) {
|
||||||
builder.withDeleted();
|
builder.withDeleted();
|
||||||
}
|
}
|
||||||
|
@ -12,12 +12,24 @@ import archiver from 'archiver';
|
|||||||
import chokidar, { WatchOptions } from 'chokidar';
|
import chokidar, { WatchOptions } from 'chokidar';
|
||||||
import { glob } from 'glob';
|
import { glob } from 'glob';
|
||||||
import { constants, createReadStream, existsSync, mkdirSync } from 'node:fs';
|
import { constants, createReadStream, existsSync, mkdirSync } from 'node:fs';
|
||||||
import fs, { copyFile, readdir, rename, utimes, writeFile } from 'node:fs/promises';
|
import fs, { copyFile, readdir, rename, stat, utimes, writeFile } from 'node:fs/promises';
|
||||||
import path from 'node:path';
|
import path from 'node:path';
|
||||||
|
|
||||||
export class FilesystemProvider implements IStorageRepository {
|
export class FilesystemProvider implements IStorageRepository {
|
||||||
private logger = new ImmichLogger(FilesystemProvider.name);
|
private logger = new ImmichLogger(FilesystemProvider.name);
|
||||||
|
|
||||||
|
readdir = readdir;
|
||||||
|
|
||||||
|
copyFile = copyFile;
|
||||||
|
|
||||||
|
stat = stat;
|
||||||
|
|
||||||
|
writeFile = writeFile;
|
||||||
|
|
||||||
|
rename = rename;
|
||||||
|
|
||||||
|
utimes = utimes;
|
||||||
|
|
||||||
createZipStream(): ImmichZipStream {
|
createZipStream(): ImmichZipStream {
|
||||||
const archive = archiver('zip', { store: true });
|
const archive = archiver('zip', { store: true });
|
||||||
|
|
||||||
@ -50,14 +62,6 @@ export class FilesystemProvider implements IStorageRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
writeFile = writeFile;
|
|
||||||
|
|
||||||
rename = rename;
|
|
||||||
|
|
||||||
copyFile = copyFile;
|
|
||||||
|
|
||||||
utimes = utimes;
|
|
||||||
|
|
||||||
async checkFileExists(filepath: string, mode = constants.F_OK): Promise<boolean> {
|
async checkFileExists(filepath: string, mode = constants.F_OK): Promise<boolean> {
|
||||||
try {
|
try {
|
||||||
await fs.access(filepath, mode);
|
await fs.access(filepath, mode);
|
||||||
@ -79,8 +83,6 @@ export class FilesystemProvider implements IStorageRepository {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
stat = fs.stat;
|
|
||||||
|
|
||||||
async unlinkDir(folder: string, options: { recursive?: boolean; force?: boolean }) {
|
async unlinkDir(folder: string, options: { recursive?: boolean; force?: boolean }) {
|
||||||
await fs.rm(folder, options);
|
await fs.rm(folder, options);
|
||||||
}
|
}
|
||||||
@ -146,6 +148,4 @@ export class FilesystemProvider implements IStorageRepository {
|
|||||||
|
|
||||||
return () => watcher.close();
|
return () => watcher.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
readdir = readdir;
|
|
||||||
}
|
}
|
||||||
|
19
server/test/fixtures/user.stub.ts
vendored
19
server/test/fixtures/user.stub.ts
vendored
@ -140,6 +140,25 @@ export const userStub = {
|
|||||||
quotaSizeInBytes: null,
|
quotaSizeInBytes: null,
|
||||||
quotaUsageInBytes: 0,
|
quotaUsageInBytes: 0,
|
||||||
}),
|
}),
|
||||||
|
externalPathRoot: Object.freeze<UserEntity>({
|
||||||
|
...authStub.user1.user,
|
||||||
|
password: 'immich_password',
|
||||||
|
name: 'immich_name',
|
||||||
|
storageLabel: 'label-1',
|
||||||
|
externalPath: '/',
|
||||||
|
oauthId: '',
|
||||||
|
shouldChangePassword: false,
|
||||||
|
profileImagePath: '',
|
||||||
|
createdAt: new Date('2021-01-01'),
|
||||||
|
deletedAt: null,
|
||||||
|
updatedAt: new Date('2021-01-01'),
|
||||||
|
tags: [],
|
||||||
|
assets: [],
|
||||||
|
memoriesEnabled: true,
|
||||||
|
avatarColor: UserAvatarColor.PRIMARY,
|
||||||
|
quotaSizeInBytes: null,
|
||||||
|
quotaUsageInBytes: 0,
|
||||||
|
}),
|
||||||
profilePath: Object.freeze<UserEntity>({
|
profilePath: Object.freeze<UserEntity>({
|
||||||
...authStub.user1.user,
|
...authStub.user1.user,
|
||||||
password: 'immich_password',
|
password: 'immich_password',
|
||||||
|
@ -1,13 +1,15 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import Icon from '$lib/components/elements/icon.svelte';
|
|
||||||
import type { LibraryResponseDto } from '@immich/sdk';
|
|
||||||
import { mdiPencilOutline } from '@mdi/js';
|
|
||||||
import { createEventDispatcher, onMount } from 'svelte';
|
import { createEventDispatcher, onMount } from 'svelte';
|
||||||
import { handleError } from '../../utils/handle-error';
|
import { handleError } from '../../utils/handle-error';
|
||||||
import Button from '../elements/buttons/button.svelte';
|
import Button from '../elements/buttons/button.svelte';
|
||||||
import LibraryImportPathForm from './library-import-path-form.svelte';
|
import LibraryImportPathForm from './library-import-path-form.svelte';
|
||||||
|
import Icon from '$lib/components/elements/icon.svelte';
|
||||||
|
import { mdiAlertOutline, mdiCheckCircleOutline, mdiPencilOutline, mdiRefresh } from '@mdi/js';
|
||||||
|
import { validate, type LibraryResponseDto } from '@immich/sdk';
|
||||||
|
import type { ValidateLibraryImportPathResponseDto } from '@immich/sdk/axios';
|
||||||
|
import { NotificationType, notificationController } from '../shared-components/notification/notification';
|
||||||
|
|
||||||
export let library: Partial<LibraryResponseDto>;
|
export let library: LibraryResponseDto;
|
||||||
|
|
||||||
let addImportPath = false;
|
let addImportPath = false;
|
||||||
let editImportPath: number | null = null;
|
let editImportPath: number | null = null;
|
||||||
@ -15,20 +17,62 @@
|
|||||||
let importPathToAdd: string | null = null;
|
let importPathToAdd: string | null = null;
|
||||||
let editedImportPath: string;
|
let editedImportPath: string;
|
||||||
|
|
||||||
let importPaths: string[] = [];
|
let validatedPaths: ValidateLibraryImportPathResponseDto[] = [];
|
||||||
|
|
||||||
onMount(() => {
|
$: importPaths = validatedPaths.map((validatedPath) => validatedPath.importPath);
|
||||||
|
|
||||||
|
onMount(async () => {
|
||||||
if (library.importPaths) {
|
if (library.importPaths) {
|
||||||
importPaths = library.importPaths;
|
await handleValidation();
|
||||||
} else {
|
} else {
|
||||||
library.importPaths = [];
|
library.importPaths = [];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const handleValidation = async () => {
|
||||||
|
if (library.importPaths) {
|
||||||
|
const validation = await validate({
|
||||||
|
id: library.id,
|
||||||
|
validateLibraryDto: { importPaths: library.importPaths },
|
||||||
|
});
|
||||||
|
|
||||||
|
validatedPaths = validation.importPaths ?? [];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const revalidate = async (notifyIfSuccessful = true) => {
|
||||||
|
await handleValidation();
|
||||||
|
let failedPaths = 0;
|
||||||
|
for (const validatedPath of validatedPaths) {
|
||||||
|
if (!validatedPath.isValid) {
|
||||||
|
failedPaths++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (failedPaths === 0) {
|
||||||
|
if (notifyIfSuccessful) {
|
||||||
|
notificationController.show({
|
||||||
|
message: `All paths validated successfully`,
|
||||||
|
type: NotificationType.Info,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else if (failedPaths === 1) {
|
||||||
|
notificationController.show({
|
||||||
|
message: `${failedPaths} path failed validation`,
|
||||||
|
type: NotificationType.Warning,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
notificationController.show({
|
||||||
|
message: `${failedPaths} paths failed validation`,
|
||||||
|
type: NotificationType.Warning,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const dispatch = createEventDispatcher<{
|
const dispatch = createEventDispatcher<{
|
||||||
cancel: void;
|
cancel: void;
|
||||||
submit: Partial<LibraryResponseDto>;
|
submit: Partial<LibraryResponseDto>;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
dispatch('cancel');
|
dispatch('cancel');
|
||||||
};
|
};
|
||||||
@ -50,7 +94,7 @@
|
|||||||
// Check so that import path isn't duplicated
|
// Check so that import path isn't duplicated
|
||||||
if (!library.importPaths.includes(importPathToAdd)) {
|
if (!library.importPaths.includes(importPathToAdd)) {
|
||||||
library.importPaths.push(importPathToAdd);
|
library.importPaths.push(importPathToAdd);
|
||||||
importPaths = library.importPaths;
|
await revalidate(false);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
handleError(error, 'Unable to add import path');
|
handleError(error, 'Unable to add import path');
|
||||||
@ -75,7 +119,7 @@
|
|||||||
if (!library.importPaths.includes(editedImportPath)) {
|
if (!library.importPaths.includes(editedImportPath)) {
|
||||||
// Update import path
|
// Update import path
|
||||||
library.importPaths[editImportPath] = editedImportPath;
|
library.importPaths[editImportPath] = editedImportPath;
|
||||||
importPaths = library.importPaths;
|
await revalidate(false);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
editImportPath = null;
|
editImportPath = null;
|
||||||
@ -97,7 +141,7 @@
|
|||||||
|
|
||||||
const pathToDelete = library.importPaths[editImportPath];
|
const pathToDelete = library.importPaths[editImportPath];
|
||||||
library.importPaths = library.importPaths.filter((path) => path != pathToDelete);
|
library.importPaths = library.importPaths.filter((path) => path != pathToDelete);
|
||||||
importPaths = library.importPaths;
|
await handleValidation();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
handleError(error, 'Unable to delete import path');
|
handleError(error, 'Unable to delete import path');
|
||||||
} finally {
|
} finally {
|
||||||
@ -138,7 +182,7 @@
|
|||||||
<form on:submit|preventDefault={() => handleSubmit()} autocomplete="off" class="m-4 flex flex-col gap-4">
|
<form on:submit|preventDefault={() => handleSubmit()} autocomplete="off" class="m-4 flex flex-col gap-4">
|
||||||
<table class="text-left">
|
<table class="text-left">
|
||||||
<tbody class="block w-full overflow-y-auto rounded-md border dark:border-immich-dark-gray">
|
<tbody class="block w-full overflow-y-auto rounded-md border dark:border-immich-dark-gray">
|
||||||
{#each importPaths as importPath, listIndex}
|
{#each validatedPaths as validatedPath, listIndex}
|
||||||
<tr
|
<tr
|
||||||
class={`flex h-[80px] w-full place-items-center text-center dark:text-immich-dark-fg ${
|
class={`flex h-[80px] w-full place-items-center text-center dark:text-immich-dark-fg ${
|
||||||
listIndex % 2 == 0
|
listIndex % 2 == 0
|
||||||
@ -146,13 +190,31 @@
|
|||||||
: 'bg-immich-bg dark:bg-immich-dark-gray/50'
|
: 'bg-immich-bg dark:bg-immich-dark-gray/50'
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<td class="w-4/5 text-ellipsis px-4 text-sm">{importPath}</td>
|
<td class="w-1/8 text-ellipsis pl-8 text-sm">
|
||||||
<td class="w-1/5 text-ellipsis px-4 text-sm">
|
{#if validatedPath.isValid}
|
||||||
|
<Icon
|
||||||
|
path={mdiCheckCircleOutline}
|
||||||
|
size="24"
|
||||||
|
title={validatedPath.message}
|
||||||
|
class="text-immich-success dark:text-immich-dark-success"
|
||||||
|
/>
|
||||||
|
{:else}
|
||||||
|
<Icon
|
||||||
|
path={mdiAlertOutline}
|
||||||
|
size="24"
|
||||||
|
title={validatedPath.message}
|
||||||
|
class="text-immich-warning dark:text-immich-dark-warning"
|
||||||
|
/>
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
|
||||||
|
<td class="w-4/5 text-ellipsis px-4 text-sm">{validatedPath.importPath}</td>
|
||||||
|
<td class="w-1/5 text-ellipsis px-4 text-sm flex flex-row">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
on:click={() => {
|
on:click={() => {
|
||||||
editImportPath = listIndex;
|
editImportPath = listIndex;
|
||||||
editedImportPath = importPath;
|
editedImportPath = validatedPath.importPath;
|
||||||
}}
|
}}
|
||||||
class="rounded-full bg-immich-primary p-3 text-gray-100 transition-all duration-150 hover:bg-immich-primary/75 dark:bg-immich-dark-primary dark:text-gray-700"
|
class="rounded-full bg-immich-primary p-3 text-gray-100 transition-all duration-150 hover:bg-immich-primary/75 dark:bg-immich-dark-primary dark:text-gray-700"
|
||||||
>
|
>
|
||||||
@ -185,9 +247,13 @@
|
|||||||
>
|
>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
<div class="flex justify-between w-full">
|
||||||
<div class="flex w-full justify-end gap-2">
|
<div class="justify-end gap-2">
|
||||||
<Button size="sm" color="gray" on:click={() => handleCancel()}>Cancel</Button>
|
<Button size="sm" color="gray" on:click={() => revalidate()}><Icon path={mdiRefresh} size={20} />Validate</Button>
|
||||||
<Button size="sm" type="submit">Save</Button>
|
</div>
|
||||||
|
<div class="justify-end gap-2">
|
||||||
|
<Button size="sm" color="gray" on:click={() => handleCancel()}>Cancel</Button>
|
||||||
|
<Button size="sm" type="submit">Save</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
@ -126,11 +126,10 @@
|
|||||||
try {
|
try {
|
||||||
const libraryId = libraries[updateLibraryIndex].id;
|
const libraryId = libraries[updateLibraryIndex].id;
|
||||||
await updateLibrary({ id: libraryId, updateLibraryDto: { ...event } });
|
await updateLibrary({ id: libraryId, updateLibraryDto: { ...event } });
|
||||||
} catch (error) {
|
|
||||||
handleError(error, 'Unable to update library');
|
|
||||||
} finally {
|
|
||||||
closeAll();
|
closeAll();
|
||||||
await readLibraryList();
|
await readLibraryList();
|
||||||
|
} catch (error) {
|
||||||
|
handleError(error, 'Unable to update library');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user