mirror of
https://github.com/immich-app/immich.git
synced 2026-04-15 15:11:54 -04:00
* feat: add a `maintenance.enabled` config flag
* feat: implement graceful restart
feat: restart when maintenance config is toggled
* feat: boot a stripped down maintenance api if enabled
* feat: cli command to toggle maintenance mode
* chore: fallback IMMICH_SERVER_URL environment variable in process
* chore: add additional routes to maintenance controller
* fix: don't wait for nest application to close to finish request response
* chore: add a failsafe on restart to prevent other exit codes from preventing restart
* feat: redirect into/from maintenance page
* refactor: use system metadata for maintenance status
* refactor: wait on WebSocket connection to refresh
* feat: broadcast websocket event on server restart
refactor: listen to WS instead of polling
* refactor: bubble up maintenance information instead of hijacking in fetch function
feat: show modal when server is restarting
* chore: increase timeout for ungraceful restart
* refactor: deduplicate code between api/maintenance workers
* fix: skip config check if database is not initialised
* fix: add `maintenanceMode` field to system config test
* refactor: move maintenance resolution code to static method in service
* chore: clean up linter issues
* chore: generate dart openapi
* refactor: use try{} block for maintenance mode check
* fix: logic error in server redirect
* chore: include `maintenanceMode` key in e2e test
* chore: add i18n entries for maintenance screens
* chore: remove negated condition from hook
* fix: should set default value not override in service
* fix: minor error in page
* feat: initial draft of maintenance module, repo., worker controller, worker service
* refactor: move broadcast code into notification service
* chore: connect websocket on client if in maintenance
* chore: set maintenance module app name
* refactor: rename repository to include worker
chore: configure websocket adapter
* feat: reimplement maintenance mode exit with new module
* refactor: add a constant enum for ExitCode
* refactor: remove redundant route for maintenance
* refactor: only spin up kysely on boot (rather than a Nest app)
* refactor(web): move redirect logic into +layout file where modal is setup
* feat: add Maintenance permission
* refactor: merge common code between api/maintenance
* fix: propagate changes from the CLI to servers
* feat: maintenance authentication guard
* refactor: unify maintenance code into repository
feat: add a step to generate maintenance mode token
* feat: jwt auth for maintenance
* refactor: switch from nest jwt to just jsonwebtokens
* feat: log into maintenance mode from CLI command
* refactor: use `secret` instead of `token` in jwt terminology
chore: log maintenance mode login URL on boot
chore: don't make CLI actions reload if already in target state
* docs: initial draft for maintenance mode page
* refactor: always validate the maintenance auth on the server
* feat: add a link to maintenance mode documentation
* feat: redirect users back to the last page they were on when exiting maintenance
* refactor: provide closeFn in both maintenance repos.
* refactor: ensure the user is also redirected by the server
* chore: swap jsonwebtoken for jose
* refactor: introduce AppRestartEvent w/o secret passing
* refactor: use navigation goto
* refactor: use `continue` instead of `next`
* chore: lint fixes for server
* chore: lint fixes for web
* test: add mock for maintenance repository
* test: add base service dependency to maintenance
* chore: remove @types/jsonwebtoken
* refactor: close database connection after startup check
* refactor: use `request#auth` key
* refactor: use service instead of repository
chore: read token from cookie if possible
chore: rename client event to AppRestartV1
* refactor: more concise redirect logic on web
* refactor: move redirect check into utils
refactor: update translation strings to be more sensible
* refactor: always validate login (i.e. check cookie)
* refactor: lint, open-api, remove old dto
* refactor: encode at point of usage
* refactor: remove business logic from repositories
* chore: fix server/web lints
* refactor: remove repository mock
* chore: fix formatting
* test: write service mocks for maintenance mode
* test: write cli service tests
* fix: catch errors when closing app
* fix: always report no maintenance when usual API is available
* test: api e2e maintenance spec
* chore: add response builder
* chore: add helper to set maint. auth cookie
* feat: add SSR to maintenance API
* test(e2e): write web spec for maintenance
* chore: clean up lint issues
* chore: format files
* feat: perform 302 redirect at server level during maintenance
* fix: keep trying to stop immich until it succeeds (CLI issue)
* chore: lint/format
* refactor: annotate references to other services in worker service
* chore: lint
* refactor: remove unnecessary await
Co-authored-by: Daniel Dietzler <36593685+danieldietzler@users.noreply.github.com>
* refactor: move static methods into util
* refactor: assert secret exists in maintenance worker
* refactor: remove assertion which isn't necessary anymore
* refactor: remove assertion
* refactor: remove outer try {} catch block from loadMaintenanceAuth
* refactor: undo earlier change to vite.config.ts
* chore: update tests due to refactors
* revert: vite.config.ts
* test: expect string jwt
* chore: move blanket exceptions into controllers
* test: update tests according with last change
* refactor: use respondWithCookie
refactor: merge start/end into one route
refactor: rename MaintenanceRepository to AppRepository
chore: use new ApiTag/Endpoint
refactor: apply other requested changes
* chore: regenerate openapi
* chore: lint/format
* chore: remove secureOnly for maint. cookie
* refactor: move maintenance worker code into src/maintenance\nfix: various test fixes
* refactor: use `action` property for setting maint. mode
* refactor: remove Websocket#restartApp in favour of individual methods
* chore: incomplete commit
* chore: remove stray log
* fix: call exitApp from maintenance worker on exit
* fix: add app repository mock
* fix: ensure maintenance cookies are secure
* fix: run playwright tests over secure context (localhost)
* test: update other references to 127.0.0.1
* refactor: use serverSideEmitWithAck
* chore: correct the logic in tryTerminate
* test: juggle cookies ourselves
* chore: fix lint error for e2e spec
* chore: format e2e test
* fix: set cookie secure/non-secure depending on context
* chore: format files
---------
Co-authored-by: Daniel Dietzler <36593685+danieldietzler@users.noreply.github.com>
580 lines
14 KiB
TypeScript
580 lines
14 KiB
TypeScript
import { SystemConfig } from 'src/config';
|
|
import { VECTOR_EXTENSIONS } from 'src/constants';
|
|
import { Asset } from 'src/database';
|
|
import { UploadFieldName } from 'src/dtos/asset-media.dto';
|
|
import { AuthDto } from 'src/dtos/auth.dto';
|
|
import {
|
|
AssetMetadataKey,
|
|
AssetOrder,
|
|
AssetType,
|
|
DatabaseSslMode,
|
|
ExifOrientation,
|
|
ImageFormat,
|
|
JobName,
|
|
MemoryType,
|
|
PluginTriggerType,
|
|
QueueName,
|
|
StorageFolder,
|
|
SyncEntityType,
|
|
SystemMetadataKey,
|
|
TranscodeTarget,
|
|
UserMetadataKey,
|
|
VideoCodec,
|
|
} from 'src/enum';
|
|
|
|
export type DeepPartial<T> = T extends object ? { [K in keyof T]?: DeepPartial<T[K]> } : T;
|
|
|
|
export type RepositoryInterface<T extends object> = Pick<T, keyof T>;
|
|
|
|
export interface CropOptions {
|
|
top: number;
|
|
left: number;
|
|
width: number;
|
|
height: number;
|
|
}
|
|
|
|
export interface FullsizeImageOptions {
|
|
format: ImageFormat;
|
|
quality: number;
|
|
enabled: boolean;
|
|
}
|
|
|
|
export interface ImageOptions {
|
|
format: ImageFormat;
|
|
quality: number;
|
|
size: number;
|
|
}
|
|
|
|
export interface RawImageInfo {
|
|
width: number;
|
|
height: number;
|
|
channels: 1 | 2 | 3 | 4;
|
|
}
|
|
|
|
interface DecodeImageOptions {
|
|
colorspace: string;
|
|
crop?: CropOptions;
|
|
processInvalidImages: boolean;
|
|
raw?: RawImageInfo;
|
|
}
|
|
|
|
export interface DecodeToBufferOptions extends DecodeImageOptions {
|
|
size?: number;
|
|
orientation?: ExifOrientation;
|
|
}
|
|
|
|
export type GenerateThumbnailOptions = Pick<ImageOptions, 'format' | 'quality'> & DecodeToBufferOptions;
|
|
|
|
export type GenerateThumbnailFromBufferOptions = GenerateThumbnailOptions & { raw: RawImageInfo };
|
|
|
|
export type GenerateThumbhashOptions = DecodeImageOptions;
|
|
|
|
export type GenerateThumbhashFromBufferOptions = GenerateThumbhashOptions & { raw: RawImageInfo };
|
|
|
|
export interface GenerateThumbnailsOptions {
|
|
colorspace: string;
|
|
crop?: CropOptions;
|
|
preview?: ImageOptions;
|
|
processInvalidImages: boolean;
|
|
thumbhash?: boolean;
|
|
thumbnail?: ImageOptions;
|
|
}
|
|
|
|
export interface VideoStreamInfo {
|
|
index: number;
|
|
height: number;
|
|
width: number;
|
|
rotation: number;
|
|
codecName?: string;
|
|
frameCount: number;
|
|
isHDR: boolean;
|
|
bitrate: number;
|
|
pixelFormat: string;
|
|
colorPrimaries?: string;
|
|
colorSpace?: string;
|
|
colorTransfer?: string;
|
|
}
|
|
|
|
export interface AudioStreamInfo {
|
|
index: number;
|
|
codecName?: string;
|
|
bitrate: number;
|
|
}
|
|
|
|
export interface VideoFormat {
|
|
formatName?: string;
|
|
formatLongName?: string;
|
|
duration: number;
|
|
bitrate: number;
|
|
}
|
|
|
|
export interface ImageDimensions {
|
|
width: number;
|
|
height: number;
|
|
}
|
|
|
|
export interface InputDimensions extends ImageDimensions {
|
|
inputPath: string;
|
|
}
|
|
|
|
export interface VideoInfo {
|
|
format: VideoFormat;
|
|
videoStreams: VideoStreamInfo[];
|
|
audioStreams: AudioStreamInfo[];
|
|
}
|
|
|
|
export interface TranscodeCommand {
|
|
inputOptions: string[];
|
|
outputOptions: string[];
|
|
twoPass: boolean;
|
|
progress: {
|
|
frameCount: number;
|
|
percentInterval: number;
|
|
};
|
|
}
|
|
|
|
export interface BitrateDistribution {
|
|
max: number;
|
|
target: number;
|
|
min: number;
|
|
unit: string;
|
|
}
|
|
|
|
export interface ImageBuffer {
|
|
data: Buffer;
|
|
info: RawImageInfo;
|
|
}
|
|
|
|
export interface VideoCodecSWConfig {
|
|
getCommand(
|
|
target: TranscodeTarget,
|
|
videoStream: VideoStreamInfo,
|
|
audioStream: AudioStreamInfo,
|
|
format?: VideoFormat,
|
|
): TranscodeCommand;
|
|
}
|
|
|
|
export interface VideoCodecHWConfig extends VideoCodecSWConfig {
|
|
getSupportedCodecs(): Array<VideoCodec>;
|
|
}
|
|
|
|
export interface ProbeOptions {
|
|
countFrames: boolean;
|
|
}
|
|
|
|
export interface VideoInterfaces {
|
|
dri: string[];
|
|
mali: boolean;
|
|
}
|
|
|
|
export type ConcurrentQueueName = Exclude<
|
|
QueueName,
|
|
| QueueName.StorageTemplateMigration
|
|
| QueueName.FacialRecognition
|
|
| QueueName.DuplicateDetection
|
|
| QueueName.BackupDatabase
|
|
>;
|
|
|
|
export type Jobs = { [K in JobItem['name']]: (JobItem & { name: K })['data'] };
|
|
export type JobOf<T extends JobName> = Jobs[T];
|
|
|
|
export interface IBaseJob {
|
|
force?: boolean;
|
|
}
|
|
|
|
export interface IDelayedJob extends IBaseJob {
|
|
/** The minimum time to wait to execute this job, in milliseconds. */
|
|
delay?: number;
|
|
}
|
|
|
|
export type JobSource = 'upload' | 'sidecar-write' | 'copy';
|
|
export interface IEntityJob extends IBaseJob {
|
|
id: string;
|
|
source?: JobSource;
|
|
notify?: boolean;
|
|
}
|
|
|
|
export interface IAssetDeleteJob extends IEntityJob {
|
|
deleteOnDisk: boolean;
|
|
}
|
|
|
|
export interface ILibraryFileJob {
|
|
libraryId: string;
|
|
paths: string[];
|
|
progressCounter?: number;
|
|
totalAssets?: number;
|
|
}
|
|
|
|
export interface ILibraryBulkIdsJob {
|
|
libraryId: string;
|
|
importPaths: string[];
|
|
exclusionPatterns: string[];
|
|
assetIds: string[];
|
|
progressCounter: number;
|
|
totalAssets: number;
|
|
}
|
|
|
|
export interface IBulkEntityJob {
|
|
ids: string[];
|
|
}
|
|
|
|
export interface IDeleteFilesJob extends IBaseJob {
|
|
files: Array<string | null | undefined>;
|
|
}
|
|
|
|
export interface ISidecarWriteJob extends IEntityJob {
|
|
description?: string;
|
|
dateTimeOriginal?: string;
|
|
latitude?: number;
|
|
longitude?: number;
|
|
rating?: number;
|
|
tags?: true;
|
|
}
|
|
|
|
export interface IDeferrableJob extends IEntityJob {
|
|
deferred?: boolean;
|
|
}
|
|
|
|
export interface INightlyJob extends IBaseJob {
|
|
nightly?: boolean;
|
|
}
|
|
|
|
export type EmailImageAttachment = {
|
|
filename: string;
|
|
path: string;
|
|
cid: string;
|
|
};
|
|
|
|
export interface IEmailJob {
|
|
to: string;
|
|
subject: string;
|
|
html: string;
|
|
text: string;
|
|
imageAttachments?: EmailImageAttachment[];
|
|
}
|
|
|
|
export interface INotifySignupJob extends IEntityJob {
|
|
password?: string;
|
|
}
|
|
|
|
export interface INotifyAlbumInviteJob extends IEntityJob {
|
|
recipientId: string;
|
|
}
|
|
|
|
export interface INotifyAlbumUpdateJob extends IEntityJob, IDelayedJob {
|
|
recipientId: string;
|
|
}
|
|
|
|
export interface WorkflowData {
|
|
[PluginTriggerType.AssetCreate]: {
|
|
userId: string;
|
|
asset: Asset;
|
|
};
|
|
[PluginTriggerType.PersonRecognized]: {
|
|
personId: string;
|
|
assetId: string;
|
|
};
|
|
}
|
|
|
|
export interface IWorkflowJob<T extends PluginTriggerType = PluginTriggerType> {
|
|
id: string;
|
|
type: T;
|
|
event: WorkflowData[T];
|
|
}
|
|
|
|
export interface JobCounts {
|
|
active: number;
|
|
completed: number;
|
|
failed: number;
|
|
delayed: number;
|
|
waiting: number;
|
|
paused: number;
|
|
}
|
|
|
|
export interface QueueStatus {
|
|
isActive: boolean;
|
|
isPaused: boolean;
|
|
}
|
|
|
|
export type JobItem =
|
|
// Audit
|
|
| { name: JobName.AuditTableCleanup; data?: IBaseJob }
|
|
|
|
// Backups
|
|
| { name: JobName.DatabaseBackup; data?: IBaseJob }
|
|
|
|
// Transcoding
|
|
| { name: JobName.AssetEncodeVideoQueueAll; data: IBaseJob }
|
|
| { name: JobName.AssetEncodeVideo; data: IEntityJob }
|
|
|
|
// Thumbnails
|
|
| { name: JobName.AssetGenerateThumbnailsQueueAll; data: IBaseJob }
|
|
| { name: JobName.AssetGenerateThumbnails; data: IEntityJob }
|
|
|
|
// User
|
|
| { name: JobName.UserDeleteCheck; data?: IBaseJob }
|
|
| { name: JobName.UserDelete; data: IEntityJob }
|
|
| { name: JobName.UserSyncUsage; data?: IBaseJob }
|
|
|
|
// Storage Template
|
|
| { name: JobName.StorageTemplateMigration; data?: IBaseJob }
|
|
| { name: JobName.StorageTemplateMigrationSingle; data: IEntityJob }
|
|
|
|
// Migration
|
|
| { name: JobName.FileMigrationQueueAll; data?: IBaseJob }
|
|
| { name: JobName.AssetFileMigration; data: IEntityJob }
|
|
| { name: JobName.PersonFileMigration; data: IEntityJob }
|
|
|
|
// Metadata Extraction
|
|
| { name: JobName.AssetExtractMetadataQueueAll; data: IBaseJob }
|
|
| { name: JobName.AssetExtractMetadata; data: IEntityJob }
|
|
|
|
// Notifications
|
|
| { name: JobName.NotificationsCleanup; data?: IBaseJob }
|
|
|
|
// Sidecar Scanning
|
|
| { name: JobName.SidecarQueueAll; data: IBaseJob }
|
|
| { name: JobName.SidecarCheck; data: IEntityJob }
|
|
| { name: JobName.SidecarWrite; data: ISidecarWriteJob }
|
|
|
|
// Facial Recognition
|
|
| { name: JobName.AssetDetectFacesQueueAll; data: IBaseJob }
|
|
| { name: JobName.AssetDetectFaces; data: IEntityJob }
|
|
| { name: JobName.FacialRecognitionQueueAll; data: INightlyJob }
|
|
| { name: JobName.FacialRecognition; data: IDeferrableJob }
|
|
| { name: JobName.PersonGenerateThumbnail; data: IEntityJob }
|
|
|
|
// Smart Search
|
|
| { name: JobName.SmartSearchQueueAll; data: IBaseJob }
|
|
| { name: JobName.SmartSearch; data: IEntityJob }
|
|
| { name: JobName.AssetEmptyTrash; data?: IBaseJob }
|
|
|
|
// Duplicate Detection
|
|
| { name: JobName.AssetDetectDuplicatesQueueAll; data: IBaseJob }
|
|
| { name: JobName.AssetDetectDuplicates; data: IEntityJob }
|
|
|
|
// Memories
|
|
| { name: JobName.MemoryCleanup; data?: IBaseJob }
|
|
| { name: JobName.MemoryGenerate; data?: IBaseJob }
|
|
|
|
// Filesystem
|
|
| { name: JobName.FileDelete; data: IDeleteFilesJob }
|
|
|
|
// Cleanup
|
|
| { name: JobName.AuditLogCleanup; data?: IBaseJob }
|
|
| { name: JobName.SessionCleanup; data?: IBaseJob }
|
|
|
|
// Tags
|
|
| { name: JobName.TagCleanup; data?: IBaseJob }
|
|
|
|
// Asset Deletion
|
|
| { name: JobName.PersonCleanup; data?: IBaseJob }
|
|
| { name: JobName.AssetDelete; data: IAssetDeleteJob }
|
|
| { name: JobName.AssetDeleteCheck; data?: IBaseJob }
|
|
|
|
// Library Management
|
|
| { name: JobName.LibrarySyncFiles; data: ILibraryFileJob }
|
|
| { name: JobName.LibrarySyncFilesQueueAll; data: IEntityJob }
|
|
| { name: JobName.LibrarySyncAssetsQueueAll; data: IEntityJob }
|
|
| { name: JobName.LibrarySyncAssets; data: ILibraryBulkIdsJob }
|
|
| { name: JobName.LibraryRemoveAsset; data: ILibraryFileJob }
|
|
| { name: JobName.LibraryDelete; data: IEntityJob }
|
|
| { name: JobName.LibraryScanQueueAll; data?: IBaseJob }
|
|
| { name: JobName.LibraryDeleteCheck; data: IBaseJob }
|
|
|
|
// Notification
|
|
| { name: JobName.SendMail; data: IEmailJob }
|
|
| { name: JobName.NotifyAlbumInvite; data: INotifyAlbumInviteJob }
|
|
| { name: JobName.NotifyAlbumUpdate; data: INotifyAlbumUpdateJob }
|
|
| { name: JobName.NotifyUserSignup; data: INotifySignupJob }
|
|
|
|
// Version check
|
|
| { name: JobName.VersionCheck; data: IBaseJob }
|
|
|
|
// OCR
|
|
| { name: JobName.OcrQueueAll; data: IBaseJob }
|
|
| { name: JobName.Ocr; data: IEntityJob }
|
|
|
|
// Workflow
|
|
| { name: JobName.WorkflowRun; data: IWorkflowJob };
|
|
|
|
export type VectorExtension = (typeof VECTOR_EXTENSIONS)[number];
|
|
|
|
export type DatabaseConnectionURL = {
|
|
connectionType: 'url';
|
|
url: string;
|
|
};
|
|
|
|
export type DatabaseConnectionParts = {
|
|
connectionType: 'parts';
|
|
host: string;
|
|
port: number;
|
|
username: string;
|
|
password: string;
|
|
database: string;
|
|
ssl?: DatabaseSslMode;
|
|
};
|
|
|
|
export type DatabaseConnectionParams = DatabaseConnectionURL | DatabaseConnectionParts;
|
|
|
|
export interface ExtensionVersion {
|
|
name: VectorExtension;
|
|
availableVersion: string | null;
|
|
installedVersion: string | null;
|
|
}
|
|
|
|
export interface VectorUpdateResult {
|
|
restartRequired: boolean;
|
|
}
|
|
|
|
export interface ImmichFile extends Express.Multer.File {
|
|
uuid: string;
|
|
/** sha1 hash of file */
|
|
checksum: Buffer;
|
|
}
|
|
|
|
export interface UploadFile {
|
|
uuid: string;
|
|
checksum: Buffer;
|
|
originalPath: string;
|
|
originalName: string;
|
|
size: number;
|
|
}
|
|
|
|
export interface UploadBody {
|
|
filename?: string;
|
|
[key: string]: unknown;
|
|
}
|
|
|
|
export type UploadRequest = {
|
|
auth: AuthDto | null;
|
|
fieldName: UploadFieldName;
|
|
file: UploadFile;
|
|
body: UploadBody;
|
|
};
|
|
|
|
export interface UploadFiles {
|
|
assetData: ImmichFile[];
|
|
sidecarData: ImmichFile[];
|
|
}
|
|
|
|
export interface IBulkAsset {
|
|
getAssetIds: (id: string, assetIds: string[]) => Promise<Set<string>>;
|
|
addAssetIds: (id: string, assetIds: string[]) => Promise<void>;
|
|
removeAssetIds: (id: string, assetIds: string[]) => Promise<void>;
|
|
}
|
|
|
|
export type SyncAck = {
|
|
type: SyncEntityType;
|
|
updateId: string;
|
|
extraId?: string;
|
|
};
|
|
|
|
export type StorageAsset = {
|
|
id: string;
|
|
ownerId: string;
|
|
livePhotoVideoId: string | null;
|
|
type: AssetType;
|
|
isExternal: boolean;
|
|
checksum: Buffer;
|
|
timeZone: string | null;
|
|
fileCreatedAt: Date;
|
|
originalPath: string;
|
|
originalFileName: string;
|
|
sidecarPath: string | null;
|
|
fileSizeInByte: number | null;
|
|
};
|
|
|
|
export type OnThisDayData = { year: number };
|
|
|
|
export interface MemoryData {
|
|
[MemoryType.OnThisDay]: OnThisDayData;
|
|
}
|
|
|
|
export type VersionCheckMetadata = { checkedAt: string; releaseVersion: string };
|
|
export type SystemFlags = { mountChecks: Record<StorageFolder, boolean> };
|
|
export type MaintenanceModeState = { isMaintenanceMode: true; secret: string } | { isMaintenanceMode: false };
|
|
export type MemoriesState = {
|
|
/** memories have already been created through this date */
|
|
lastOnThisDayDate: string;
|
|
};
|
|
export type MediaLocation = { location: string };
|
|
|
|
export interface SystemMetadata extends Record<SystemMetadataKey, Record<string, any>> {
|
|
[SystemMetadataKey.AdminOnboarding]: { isOnboarded: boolean };
|
|
[SystemMetadataKey.FacialRecognitionState]: { lastRun?: string };
|
|
[SystemMetadataKey.License]: { licenseKey: string; activationKey: string; activatedAt: Date };
|
|
[SystemMetadataKey.MaintenanceMode]: MaintenanceModeState;
|
|
[SystemMetadataKey.MediaLocation]: MediaLocation;
|
|
[SystemMetadataKey.ReverseGeocodingState]: { lastUpdate?: string; lastImportFileName?: string };
|
|
[SystemMetadataKey.SystemConfig]: DeepPartial<SystemConfig>;
|
|
[SystemMetadataKey.SystemFlags]: DeepPartial<SystemFlags>;
|
|
[SystemMetadataKey.VersionCheckState]: VersionCheckMetadata;
|
|
[SystemMetadataKey.MemoriesState]: MemoriesState;
|
|
}
|
|
|
|
export interface UserPreferences {
|
|
albums: {
|
|
defaultAssetOrder: AssetOrder;
|
|
};
|
|
folders: {
|
|
enabled: boolean;
|
|
sidebarWeb: boolean;
|
|
};
|
|
memories: {
|
|
enabled: boolean;
|
|
duration: number;
|
|
};
|
|
people: {
|
|
enabled: boolean;
|
|
sidebarWeb: boolean;
|
|
};
|
|
ratings: {
|
|
enabled: boolean;
|
|
};
|
|
sharedLinks: {
|
|
enabled: boolean;
|
|
sidebarWeb: boolean;
|
|
};
|
|
tags: {
|
|
enabled: boolean;
|
|
sidebarWeb: boolean;
|
|
};
|
|
emailNotifications: {
|
|
enabled: boolean;
|
|
albumInvite: boolean;
|
|
albumUpdate: boolean;
|
|
};
|
|
download: {
|
|
archiveSize: number;
|
|
includeEmbeddedVideos: boolean;
|
|
};
|
|
purchase: {
|
|
showSupportBadge: boolean;
|
|
hideBuyButtonUntil: string;
|
|
};
|
|
cast: {
|
|
gCastEnabled: boolean;
|
|
};
|
|
}
|
|
|
|
export type UserMetadataItem<T extends keyof UserMetadata = UserMetadataKey> = {
|
|
key: T;
|
|
value: UserMetadata[T];
|
|
};
|
|
|
|
export interface UserMetadata extends Record<UserMetadataKey, Record<string, any>> {
|
|
[UserMetadataKey.Preferences]: DeepPartial<UserPreferences>;
|
|
[UserMetadataKey.License]: { licenseKey: string; activationKey: string; activatedAt: string };
|
|
[UserMetadataKey.Onboarding]: { isOnboarded: boolean };
|
|
}
|
|
|
|
export type AssetMetadataItem<T extends keyof AssetMetadata = AssetMetadataKey> = {
|
|
key: T;
|
|
value: AssetMetadata[T];
|
|
};
|
|
|
|
export interface AssetMetadata extends Record<AssetMetadataKey, Record<string, any>> {
|
|
[AssetMetadataKey.MobileApp]: { iCloudId: string };
|
|
}
|