2024-02-14 15:38:57 +01:00
|
|
|
import { NotificationType, notificationController } from '$lib/components/shared-components/notification/notification';
|
2024-02-22 15:36:14 +01:00
|
|
|
import { locales } from '$lib/constants';
|
2024-02-14 15:38:57 +01:00
|
|
|
import { handleError } from '$lib/utils/handle-error';
|
|
|
|
import {
|
|
|
|
AssetJobName,
|
|
|
|
JobName,
|
|
|
|
ThumbnailFormat,
|
|
|
|
finishOAuth,
|
2024-05-17 22:48:29 +02:00
|
|
|
getBaseUrl,
|
2024-02-14 15:38:57 +01:00
|
|
|
linkOAuthAccount,
|
|
|
|
startOAuth,
|
|
|
|
unlinkOAuthAccount,
|
2024-04-03 16:37:03 +02:00
|
|
|
type SharedLinkResponseDto,
|
2024-02-14 15:38:57 +01:00
|
|
|
type UserResponseDto,
|
|
|
|
} from '@immich/sdk';
|
2024-03-21 19:39:33 +01:00
|
|
|
import { mdiCogRefreshOutline, mdiDatabaseRefreshOutline, mdiImageRefreshOutline } from '@mdi/js';
|
2024-02-14 14:09:49 +01:00
|
|
|
|
2024-02-29 17:22:39 +01:00
|
|
|
interface DownloadRequestOptions<T = unknown> {
|
|
|
|
method?: 'GET' | 'POST' | 'PUT' | 'DELETE';
|
|
|
|
url: string;
|
|
|
|
data?: T;
|
|
|
|
signal?: AbortSignal;
|
|
|
|
onDownloadProgress?: (event: ProgressEvent<XMLHttpRequestEventTarget>) => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface UploadRequestOptions {
|
|
|
|
url: string;
|
2024-05-24 02:26:22 +02:00
|
|
|
method?: 'POST' | 'PUT';
|
2024-02-29 17:22:39 +01:00
|
|
|
data: FormData;
|
|
|
|
onUploadProgress?: (event: ProgressEvent<XMLHttpRequestEventTarget>) => void;
|
|
|
|
}
|
|
|
|
|
2024-03-20 20:40:41 +01:00
|
|
|
export class AbortError extends Error {
|
2024-02-29 17:22:39 +01:00
|
|
|
name = 'AbortError';
|
|
|
|
}
|
|
|
|
|
|
|
|
class ApiError extends Error {
|
|
|
|
name = 'ApiError';
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
public message: string,
|
|
|
|
public statusCode: number,
|
|
|
|
public details: string,
|
|
|
|
) {
|
|
|
|
super(message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const uploadRequest = async <T>(options: UploadRequestOptions): Promise<{ data: T; status: number }> => {
|
|
|
|
const { onUploadProgress: onProgress, data, url } = options;
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
|
|
|
|
xhr.addEventListener('error', (error) => reject(error));
|
|
|
|
xhr.addEventListener('load', () => {
|
|
|
|
if (xhr.readyState === 4 && xhr.status >= 200 && xhr.status < 300) {
|
|
|
|
resolve({ data: xhr.response as T, status: xhr.status });
|
|
|
|
} else {
|
|
|
|
reject(new ApiError(xhr.statusText, xhr.status, xhr.response));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (onProgress) {
|
2024-03-12 12:19:38 +01:00
|
|
|
xhr.upload.addEventListener('progress', (event) => onProgress(event));
|
2024-02-29 17:22:39 +01:00
|
|
|
}
|
|
|
|
|
2024-05-24 02:26:22 +02:00
|
|
|
xhr.open(options.method || 'POST', url);
|
2024-02-29 17:22:39 +01:00
|
|
|
xhr.responseType = 'json';
|
|
|
|
xhr.send(data);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
export const downloadRequest = <TBody = unknown>(options: DownloadRequestOptions<TBody> | string) => {
|
|
|
|
if (typeof options === 'string') {
|
|
|
|
options = { url: options };
|
|
|
|
}
|
|
|
|
|
|
|
|
const { signal, method, url, data: body, onDownloadProgress: onProgress } = options;
|
|
|
|
|
|
|
|
return new Promise<{ data: Blob; status: number }>((resolve, reject) => {
|
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
|
|
|
|
xhr.addEventListener('error', (error) => reject(error));
|
|
|
|
xhr.addEventListener('abort', () => reject(new AbortError()));
|
|
|
|
xhr.addEventListener('load', () => {
|
|
|
|
if (xhr.readyState === 4 && xhr.status >= 200 && xhr.status < 300) {
|
|
|
|
resolve({ data: xhr.response as Blob, status: xhr.status });
|
|
|
|
} else {
|
|
|
|
reject(new ApiError(xhr.statusText, xhr.status, xhr.responseText));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (onProgress) {
|
|
|
|
xhr.addEventListener('progress', (event) => onProgress(event));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signal) {
|
|
|
|
signal.addEventListener('abort', () => xhr.abort());
|
|
|
|
}
|
|
|
|
|
|
|
|
xhr.open(method || 'GET', url);
|
|
|
|
xhr.responseType = 'blob';
|
|
|
|
|
|
|
|
if (body) {
|
|
|
|
xhr.setRequestHeader('Content-Type', 'application/json');
|
|
|
|
xhr.send(JSON.stringify(body));
|
|
|
|
} else {
|
|
|
|
xhr.send();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2024-02-14 14:09:49 +01:00
|
|
|
export const getJobName = (jobName: JobName) => {
|
|
|
|
const names: Record<JobName, string> = {
|
|
|
|
[JobName.ThumbnailGeneration]: 'Generate Thumbnails',
|
|
|
|
[JobName.MetadataExtraction]: 'Extract Metadata',
|
|
|
|
[JobName.Sidecar]: 'Sidecar Metadata',
|
|
|
|
[JobName.SmartSearch]: 'Smart Search',
|
2024-05-16 19:08:37 +02:00
|
|
|
[JobName.DuplicateDetection]: 'Duplicate Detection',
|
2024-02-14 14:09:49 +01:00
|
|
|
[JobName.FaceDetection]: 'Face Detection',
|
|
|
|
[JobName.FacialRecognition]: 'Facial Recognition',
|
|
|
|
[JobName.VideoConversion]: 'Transcode Videos',
|
|
|
|
[JobName.StorageTemplateMigration]: 'Storage Template Migration',
|
|
|
|
[JobName.Migration]: 'Migration',
|
|
|
|
[JobName.BackgroundTask]: 'Background Tasks',
|
|
|
|
[JobName.Search]: 'Search',
|
|
|
|
[JobName.Library]: 'Library',
|
2024-05-02 16:43:18 +02:00
|
|
|
[JobName.Notifications]: 'Notifications',
|
2024-02-14 14:09:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return names[jobName];
|
|
|
|
};
|
|
|
|
|
|
|
|
let _key: string | undefined;
|
2024-04-03 16:37:03 +02:00
|
|
|
let _sharedLink: SharedLinkResponseDto | undefined;
|
2024-02-14 14:09:49 +01:00
|
|
|
|
2024-04-03 16:37:03 +02:00
|
|
|
export const setKey = (key: string) => (_key = key);
|
|
|
|
export const getKey = (): string | undefined => _key;
|
|
|
|
export const setSharedLink = (sharedLink: SharedLinkResponseDto) => (_sharedLink = sharedLink);
|
|
|
|
export const getSharedLink = (): SharedLinkResponseDto | undefined => _sharedLink;
|
2024-02-14 14:09:49 +01:00
|
|
|
|
|
|
|
export const isSharedLink = () => {
|
|
|
|
return !!_key;
|
|
|
|
};
|
|
|
|
|
|
|
|
const createUrl = (path: string, parameters?: Record<string, unknown>) => {
|
|
|
|
const searchParameters = new URLSearchParams();
|
|
|
|
for (const key in parameters) {
|
|
|
|
const value = parameters[key];
|
|
|
|
if (value !== undefined && value !== null) {
|
|
|
|
searchParameters.set(key, value.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const url = new URL(path, 'https://example.com');
|
|
|
|
url.search = searchParameters.toString();
|
|
|
|
|
2024-05-17 22:48:29 +02:00
|
|
|
return getBaseUrl() + url.pathname + url.search + url.hash;
|
2024-02-14 14:09:49 +01:00
|
|
|
};
|
|
|
|
|
2024-05-24 02:26:22 +02:00
|
|
|
export const getAssetFileUrl = (
|
|
|
|
...[assetId, isWeb, isThumb, checksum]:
|
|
|
|
| [assetId: string, isWeb: boolean, isThumb: boolean]
|
|
|
|
| [assetId: string, isWeb: boolean, isThumb: boolean, checksum: string]
|
|
|
|
) => {
|
2024-02-14 14:09:49 +01:00
|
|
|
const path = `/asset/file/${assetId}`;
|
2024-05-24 02:26:22 +02:00
|
|
|
return createUrl(path, { isThumb, isWeb, key: getKey(), c: checksum });
|
2024-02-14 14:09:49 +01:00
|
|
|
};
|
|
|
|
|
2024-05-24 02:26:22 +02:00
|
|
|
export const getAssetThumbnailUrl = (
|
|
|
|
...[assetId, format, checksum]:
|
|
|
|
| [assetId: string, format: ThumbnailFormat | undefined]
|
|
|
|
| [assetId: string, format: ThumbnailFormat | undefined, checksum: string]
|
|
|
|
) => {
|
|
|
|
// checksum (optional) is used as a cache-buster param, since thumbs are
|
|
|
|
// served with static resource cache headers
|
2024-02-14 14:09:49 +01:00
|
|
|
const path = `/asset/thumbnail/${assetId}`;
|
2024-05-24 02:26:22 +02:00
|
|
|
return createUrl(path, { format, key: getKey(), c: checksum });
|
2024-02-14 14:09:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
export const getProfileImageUrl = (...[userId]: [string]) => {
|
2024-05-24 02:26:22 +02:00
|
|
|
const path = `/users/profile-image/${userId}`;
|
2024-02-14 14:09:49 +01:00
|
|
|
return createUrl(path);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getPeopleThumbnailUrl = (personId: string) => {
|
2024-05-22 19:24:57 +02:00
|
|
|
const path = `/people/${personId}/thumbnail`;
|
2024-02-14 14:09:49 +01:00
|
|
|
return createUrl(path);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getAssetJobName = (job: AssetJobName) => {
|
|
|
|
const names: Record<AssetJobName, string> = {
|
|
|
|
[AssetJobName.RefreshMetadata]: 'Refresh metadata',
|
|
|
|
[AssetJobName.RegenerateThumbnail]: 'Refresh thumbnails',
|
|
|
|
[AssetJobName.TranscodeVideo]: 'Refresh encoded videos',
|
|
|
|
};
|
|
|
|
|
|
|
|
return names[job];
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getAssetJobMessage = (job: AssetJobName) => {
|
|
|
|
const messages: Record<AssetJobName, string> = {
|
|
|
|
[AssetJobName.RefreshMetadata]: 'Refreshing metadata',
|
|
|
|
[AssetJobName.RegenerateThumbnail]: `Regenerating thumbnails`,
|
|
|
|
[AssetJobName.TranscodeVideo]: `Refreshing encoded video`,
|
|
|
|
};
|
|
|
|
|
|
|
|
return messages[job];
|
|
|
|
};
|
2024-02-14 15:38:57 +01:00
|
|
|
|
2024-03-21 19:39:33 +01:00
|
|
|
export const getAssetJobIcon = (job: AssetJobName) => {
|
|
|
|
const names: Record<AssetJobName, string> = {
|
|
|
|
[AssetJobName.RefreshMetadata]: mdiDatabaseRefreshOutline,
|
|
|
|
[AssetJobName.RegenerateThumbnail]: mdiImageRefreshOutline,
|
|
|
|
[AssetJobName.TranscodeVideo]: mdiCogRefreshOutline,
|
|
|
|
};
|
|
|
|
|
|
|
|
return names[job];
|
|
|
|
};
|
|
|
|
|
2024-02-14 15:38:57 +01:00
|
|
|
export const copyToClipboard = async (secret: string) => {
|
|
|
|
try {
|
|
|
|
await navigator.clipboard.writeText(secret);
|
|
|
|
notificationController.show({ message: 'Copied to clipboard!', type: NotificationType.Info });
|
|
|
|
} catch (error) {
|
|
|
|
handleError(error, 'Cannot copy to clipboard, make sure you are accessing the page through https');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const makeSharedLinkUrl = (externalDomain: string, key: string) => {
|
|
|
|
let url = externalDomain || window.location.origin;
|
|
|
|
if (!url.endsWith('/')) {
|
|
|
|
url += '/';
|
|
|
|
}
|
|
|
|
return `${url}share/${key}`;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const oauth = {
|
|
|
|
isCallback: (location: Location) => {
|
|
|
|
const search = location.search;
|
|
|
|
return search.includes('code=') || search.includes('error=');
|
|
|
|
},
|
|
|
|
isAutoLaunchDisabled: (location: Location) => {
|
|
|
|
const values = ['autoLaunch=0', 'password=1', 'password=true'];
|
|
|
|
for (const value of values) {
|
|
|
|
if (location.search.includes(value)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
authorize: async (location: Location) => {
|
|
|
|
try {
|
|
|
|
const redirectUri = location.href.split('?')[0];
|
|
|
|
const { url } = await startOAuth({ oAuthConfigDto: { redirectUri } });
|
|
|
|
window.location.href = url;
|
|
|
|
return true;
|
|
|
|
} catch (error) {
|
|
|
|
handleError(error, 'Unable to login with OAuth');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
login: (location: Location) => {
|
|
|
|
return finishOAuth({ oAuthCallbackDto: { url: location.href } });
|
|
|
|
},
|
|
|
|
link: (location: Location): Promise<UserResponseDto> => {
|
|
|
|
return linkOAuthAccount({ oAuthCallbackDto: { url: location.href } });
|
|
|
|
},
|
|
|
|
unlink: () => {
|
|
|
|
return unlinkOAuthAccount();
|
|
|
|
},
|
|
|
|
};
|
2024-02-22 15:36:14 +01:00
|
|
|
|
|
|
|
export const findLocale = (code: string | undefined) => {
|
|
|
|
const language = locales.find((lang) => lang.code === code);
|
|
|
|
return {
|
|
|
|
code: language?.code,
|
|
|
|
name: language?.name,
|
|
|
|
};
|
|
|
|
};
|
2024-02-27 17:37:37 +01:00
|
|
|
|
|
|
|
export const asyncTimeout = (ms: number) => {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
setTimeout(resolve, ms);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
export const handlePromiseError = <T>(promise: Promise<T>): void => {
|
|
|
|
promise.catch((error) => console.error(`[utils.ts]:handlePromiseError ${error}`, error));
|
|
|
|
};
|
2024-03-27 21:14:29 +01:00
|
|
|
|
2024-05-22 15:33:37 +02:00
|
|
|
export const s = (count: number) => (count === 1 ? '' : 's');
|
|
|
|
|
|
|
|
export const memoryLaneTitle = (yearsAgo: number) => `year${s(yearsAgo)} ago`;
|