mirror of
https://github.com/immich-app/immich.git
synced 2026-01-20 16:43:16 -08:00
* feat: ProcessRepository#createSpawnDuplexStream
* test: write tests for ProcessRepository#createSpawnDuplexStream
* feat: StorageRepository#createGzip,createGunzip,createPlainReadStream
* feat: backups util (args, create, restore, progress)
* feat: wait on maintenance operation lock on boot
* chore: use backup util from backup.service.ts
test: update backup.service.ts tests with new util
* feat: list/delete backups (maintenance services)
* chore: open api
fix: missing action in cli.service.ts
* chore: add missing repositories to MaintenanceModule
* refactor: move logSecret into module init
* feat: initialise StorageCore in maintenance mode
* feat: authenticate websocket requests in maintenance mode
* test: add mock for new storage fns
* feat: add MaintenanceEphemeralStateRepository
refactor: cache the secret in memory
* test: update service worker tests
* feat: add external maintenance mode status
* feat: synchronised status, restore db action
* test: backup restore service tests
* refactor: DRY end maintenance
* feat: list and delete backup routes
* feat: start action on boot
* fix: should set status on restore end
* refactor: add maintenanceStore to hold writables
* feat: sync status to web app
* feat: web impl.
* test: various utils for testings
* test: web e2e tests
* test: e2e maintenance spec
* test: update cli spec
* chore: e2e lint
* chore: lint fixes
* chore: lint fixes
* feat: start restore flow route
* test: update e2e tests
* chore: remove neon lights on maintenance action pages
* fix: use 'startRestoreFlow' on onboarding page
* chore: ignore any library folder in `docker/`
* fix: load status on boot
* feat: upload backups
* refactor: permit any .sql(.gz) to be listed/restored
* feat: download backups from list
* fix: permit uploading just .sql files
* feat: restore just .sql files
* fix: don't show backups list if logged out
* feat: system integrity check in restore flow
* test: not providing failed backups in API anymore
* test: util should also not try to use failedBackups
* fix: actually assign inputStream
* test: correct test backup prep.
* fix: ensure task is defined to show error
* test: fix docker cp command
* test: update e2e web spec to select next button
* test: update e2e api tests
* test: refactor timeouts
* chore: remove `showDelete` from maint. settings
* chore: lint
* chore: lint
* fix: make sure backups are correctly sorted for clean up
* test: update service spec
* test: adjust e2e timeout
* test: increase web timeouts for ci
* chore: move gitignore changes
* chore: additional filename validation
* refactor: better typings for integrity API
* feat: higher accuracy progress tracking
* chore: delay lock retry
* refactor: remove old maintenance settings
* refactor: clean up tailwind classes
* refactor: use while loop rather than recursive calls
* test: update service specs
* chore: check canParse too
* chore: lint
* fix: logic error causing infinite loop
* refactor: use <ProgressBar /> from ui library
* fix: create or overwrite file
* chore: i18n pass, update progress bar
* fix: wrong translation string
* chore: update colour variables
* test: update web test for new maint. page
* chore: format, fix key
* test: update tests to be more linter complaint & use new routines
* chore: update onClick -> onAction, title -> breadcrumbs
* fix: use wrench icon in admin settings sidebar
* chore: add translation strings to accordion
* chore: lint
* refactor: move maintenance worker init into service
* refactor: `maintenanceStatus` -> `getMaintenanceStatus`
refactor: `integrityCheck` -> `detectPriorInstall`
chore: add `v2.4.0` version
refactor: `/backups/list` -> `/backups`
refactor: use sendFile in download route
refactor: use separate backups permissions
chore: correct descriptions
refactor: permit handler that doesn't return promise for sendfile
* refactor: move status impl into service
refactor: add active flag to maintenance status
* refactor: split into database backup controller
* test: split api e2e tests and passing
* fix: move end button into authed default maint page
* fix: also show in restore flow
* fix: import getMaintenanceStatus
* test: split web e2e tests
* refactor: ensure detect install is consistently named
* chore: ensure admin for detect install while out of maint.
* refactor: remove state repository
* test: update maint. worker service spec
* test: split backup service spec
* refactor: rename db backup routes
* refactor: instead of param, allow bulk backup deletion
* test: update sdk use in e2e test
* test: correct deleteBackup call
* fix: correct type for serverinstall response dto
* chore: validate filename for deletion
* test: wip
* test: backups no longer take path param
* refactor: scope util to database-backups instead of backups
* fix: update worker controller with new route
* chore: use new admin page actions
* chore: remove stray comment
* test: rename outdated test
* refactor: getter pattern for maintenance secret
* refactor: `createSpawnDuplexStream` -> `spawnDuplexStream`
* refactor: prefer `Object.assign`
* refactor: remove useless try {} block
* refactor: prefer `type Props`
refactor: prefer arrow function
* refactor: use luxon API for minutesAgo
* chore: remove change to gitignore
* refactor: prefer `type Props`
* refactor: remove async from onMount
* refactor: use luxon toRelative for relative time
* refactor: duplicate logic check
* chore: open api
* refactor: begin moving code into web//services
* refactor: don't use template string with $t
* test: use dialog role to match prompt
* refactor: split actions into flow/restore
* test: fix action value
* refactor: move more service calls into web//services
* chore: should void fn return
* chore: bump 2.4.0 to 2.5.0 in controller
* chore: bump 2.4.0 to 2.5.0 in controller
* refactor: use events for web//services
* chore: open api
* chore: open api
* refactor: don't await returned promise
* refactor: remove redundant check
* refactor: add `type: command` to actions
* refactor: split backup entries into own component
* refactor: split restore flow into separate components
* refactor(web): split BackupDelete event
* chore: stylings
* chore: stylings
* fix: don't log query failure on first boot
* feat: support pg_dumpall backups
* feat: display information about each backup
* chore: i18n
* feat: rollback to restore point on migrations failure
* feat: health check after restore
* chore: format
* refactor: split health check into separate function
* refactor: split health into repository
test: write tests covering rollbacks
* fix: omit 'health' requirement from createDbBackup
* test(e2e): rollback test
* fix: wrap text in backup entry
* fix: don't shrink context menu button
* fix: correct CREATE DB syntax for postgres
* test: rename backups generated by test
* feat: add filesize to backup response dto
* feat: restore list
* feat: ui work
* fix: e2e test
* fix: e2e test
* pr feedback
* pr feedback
---------
Co-authored-by: Alex <alex.tran1502@gmail.com>
Co-authored-by: Jason Rasmussen <jason@rasm.me>
240 lines
7.8 KiB
Dart
Generated
240 lines
7.8 KiB
Dart
Generated
//
|
|
// AUTO-GENERATED FILE, DO NOT MODIFY!
|
|
//
|
|
// @dart=2.18
|
|
|
|
// ignore_for_file: unused_element, unused_import
|
|
// ignore_for_file: always_put_required_named_parameters_first
|
|
// ignore_for_file: constant_identifier_names
|
|
// ignore_for_file: lines_longer_than_80_chars
|
|
|
|
part of openapi.api;
|
|
|
|
class QueryParam {
|
|
const QueryParam(this.name, this.value);
|
|
|
|
final String name;
|
|
final String value;
|
|
|
|
@override
|
|
String toString() => '${Uri.encodeQueryComponent(name)}=${Uri.encodeQueryComponent(value)}';
|
|
}
|
|
|
|
// Ported from the Java version.
|
|
Iterable<QueryParam> _queryParams(String collectionFormat, String name, dynamic value,) {
|
|
// Assertions to run in debug mode only.
|
|
assert(name.isNotEmpty, 'Parameter cannot be an empty string.');
|
|
|
|
final params = <QueryParam>[];
|
|
|
|
if (value is List) {
|
|
if (collectionFormat == 'multi') {
|
|
return value.map((dynamic v) => QueryParam(name, parameterToString(v)),);
|
|
}
|
|
|
|
// Default collection format is 'csv'.
|
|
if (collectionFormat.isEmpty) {
|
|
collectionFormat = 'csv'; // ignore: parameter_assignments
|
|
}
|
|
|
|
final delimiter = _delimiters[collectionFormat] ?? ',';
|
|
|
|
params.add(QueryParam(name, value.map<dynamic>(parameterToString).join(delimiter),));
|
|
} else if (value != null) {
|
|
params.add(QueryParam(name, parameterToString(value)));
|
|
}
|
|
|
|
return params;
|
|
}
|
|
|
|
/// Format the given parameter object into a [String].
|
|
String parameterToString(dynamic value) {
|
|
if (value == null) {
|
|
return '';
|
|
}
|
|
if (value is DateTime) {
|
|
return value.toUtc().toIso8601String();
|
|
}
|
|
if (value is AlbumUserRole) {
|
|
return AlbumUserRoleTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetEditAction) {
|
|
return AssetEditActionTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetJobName) {
|
|
return AssetJobNameTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetMediaSize) {
|
|
return AssetMediaSizeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetMediaStatus) {
|
|
return AssetMediaStatusTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetOrder) {
|
|
return AssetOrderTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetTypeEnum) {
|
|
return AssetTypeEnumTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AssetVisibility) {
|
|
return AssetVisibilityTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is AudioCodec) {
|
|
return AudioCodecTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is BulkIdErrorReason) {
|
|
return BulkIdErrorReasonTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is CQMode) {
|
|
return CQModeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is Colorspace) {
|
|
return ColorspaceTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is ImageFormat) {
|
|
return ImageFormatTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is JobName) {
|
|
return JobNameTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is LogLevel) {
|
|
return LogLevelTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is MaintenanceAction) {
|
|
return MaintenanceActionTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is ManualJobName) {
|
|
return ManualJobNameTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is MemorySearchOrder) {
|
|
return MemorySearchOrderTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is MemoryType) {
|
|
return MemoryTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is MirrorAxis) {
|
|
return MirrorAxisTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is NotificationLevel) {
|
|
return NotificationLevelTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is NotificationType) {
|
|
return NotificationTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is OAuthTokenEndpointAuthMethod) {
|
|
return OAuthTokenEndpointAuthMethodTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is PartnerDirection) {
|
|
return PartnerDirectionTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is Permission) {
|
|
return PermissionTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is PluginContextType) {
|
|
return PluginContextTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is PluginTriggerType) {
|
|
return PluginTriggerTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is QueueCommand) {
|
|
return QueueCommandTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is QueueJobStatus) {
|
|
return QueueJobStatusTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is QueueName) {
|
|
return QueueNameTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is ReactionLevel) {
|
|
return ReactionLevelTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is ReactionType) {
|
|
return ReactionTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is SearchSuggestionType) {
|
|
return SearchSuggestionTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is SharedLinkType) {
|
|
return SharedLinkTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is SourceType) {
|
|
return SourceTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is StorageFolder) {
|
|
return StorageFolderTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is SyncEntityType) {
|
|
return SyncEntityTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is SyncRequestType) {
|
|
return SyncRequestTypeTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is ToneMapping) {
|
|
return ToneMappingTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is TranscodeHWAccel) {
|
|
return TranscodeHWAccelTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is TranscodePolicy) {
|
|
return TranscodePolicyTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is UserAvatarColor) {
|
|
return UserAvatarColorTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is UserMetadataKey) {
|
|
return UserMetadataKeyTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is UserStatus) {
|
|
return UserStatusTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is VideoCodec) {
|
|
return VideoCodecTypeTransformer().encode(value).toString();
|
|
}
|
|
if (value is VideoContainer) {
|
|
return VideoContainerTypeTransformer().encode(value).toString();
|
|
}
|
|
return value.toString();
|
|
}
|
|
|
|
/// Returns the decoded body as UTF-8 if the given headers indicate an 'application/json'
|
|
/// content type. Otherwise, returns the decoded body as decoded by dart:http package.
|
|
Future<String> _decodeBodyBytes(Response response) async {
|
|
final contentType = response.headers['content-type'];
|
|
return contentType != null && contentType.toLowerCase().startsWith('application/json')
|
|
? response.bodyBytes.isEmpty ? '' : utf8.decode(response.bodyBytes)
|
|
: response.body;
|
|
}
|
|
|
|
/// Returns a valid [T] value found at the specified Map [key], null otherwise.
|
|
T? mapValueOfType<T>(dynamic map, String key) {
|
|
final dynamic value = map is Map ? map[key] : null;
|
|
return value is T ? value : null;
|
|
}
|
|
|
|
/// Returns a valid Map<K, V> found at the specified Map [key], null otherwise.
|
|
Map<K, V>? mapCastOfType<K, V>(dynamic map, String key) {
|
|
final dynamic value = map is Map ? map[key] : null;
|
|
return value is Map ? value.cast<K, V>() : null;
|
|
}
|
|
|
|
/// Returns a valid [DateTime] found at the specified Map [key], null otherwise.
|
|
DateTime? mapDateTime(dynamic map, String key, [String? pattern]) {
|
|
final dynamic value = map is Map ? map[key] : null;
|
|
if (value != null) {
|
|
int? millis;
|
|
if (value is int) {
|
|
millis = value;
|
|
} else if (value is String) {
|
|
if (_isEpochMarker(pattern)) {
|
|
millis = int.tryParse(value);
|
|
} else {
|
|
return DateTime.tryParse(value);
|
|
}
|
|
}
|
|
if (millis != null) {
|
|
return DateTime.fromMillisecondsSinceEpoch(millis, isUtc: true);
|
|
}
|
|
}
|
|
return null;
|
|
}
|