ente/src/services/fileService.ts

270 lines
7.3 KiB
TypeScript
Raw Normal View History

2021-05-30 16:56:48 +00:00
import { getEndpoint } from 'utils/common/apiUtil';
2021-03-12 06:58:27 +00:00
import localForage from 'utils/storage/localForage';
2021-05-30 16:56:48 +00:00
import { getToken } from 'utils/common/key';
import { DataStream, MetadataObject } from './upload/uploadService';
2021-05-30 16:56:48 +00:00
import { Collection } from './collectionService';
2021-05-29 06:27:52 +00:00
import HTTPService from './HTTPService';
2021-06-12 17:14:21 +00:00
import { logError } from 'utils/sentry';
2021-07-25 07:56:05 +00:00
import { decryptFile, sortFiles } from 'utils/file';
2020-09-27 17:18:57 +00:00
const ENDPOINT = getEndpoint();
2021-07-09 08:47:30 +00:00
const DIFF_LIMIT: number = 1000;
2021-02-09 05:43:05 +00:00
const FILES = 'files';
2020-10-19 03:01:34 +00:00
export interface fileAttribute {
2021-03-04 12:14:45 +00:00
encryptedData?: DataStream | Uint8Array;
2021-02-16 09:44:25 +00:00
objectKey?: string;
decryptionHeader: string;
}
2020-10-19 03:01:34 +00:00
2021-08-13 03:19:48 +00:00
export enum FILE_TYPE {
IMAGE,
VIDEO,
LIVE_PHOTO,
OTHERS,
}
/* Build error occurred
ReferenceError: Cannot access 'FILE_TYPE' before initialization
when it was placed in readFileService
*/
// list of format that were missed by type-detection for some files.
export const FORMAT_MISSED_BY_FILE_TYPE_LIB = [
{ fileType: FILE_TYPE.IMAGE, exactType: 'jpeg' },
{ fileType: FILE_TYPE.IMAGE, exactType: 'jpg' },
{ fileType: FILE_TYPE.VIDEO, exactType: 'webm' },
];
2021-09-21 07:21:26 +00:00
export enum VISIBILITY_STATE {
VISIBLE,
ARCHIVED,
}
2021-09-21 10:37:53 +00:00
export interface MagicMetadataProps {
visibility?: VISIBILITY_STATE;
2021-09-21 07:21:26 +00:00
}
2021-09-21 10:37:53 +00:00
export interface MagicMetadata {
2021-09-21 07:21:26 +00:00
version: number;
count: number;
2021-09-21 07:45:17 +00:00
data: string | MagicMetadataProps;
2021-09-21 07:21:26 +00:00
header: string;
}
export interface File {
id: number;
collectionID: number;
file: fileAttribute;
thumbnail: fileAttribute;
2021-02-16 09:44:25 +00:00
metadata: MetadataObject;
2021-09-21 07:21:26 +00:00
magicMetadata: MagicMetadata;
encryptedKey: string;
keyDecryptionNonce: string;
2021-01-18 02:26:34 +00:00
key: string;
src: string;
msrc: string;
html: string;
w: number;
h: number;
isDeleted: boolean;
dataIndex: number;
updationTime: number;
}
2021-09-21 10:37:53 +00:00
interface UpdateMagicMetadataRequest {
metadataList: UpdateMagicMetadata[];
}
interface UpdateMagicMetadata {
id: number;
magicMetadata: MagicMetadata;
}
export const NEW_MAGIC_METADATA: MagicMetadata = {
version: 0,
data: {},
header: null,
count: 0,
};
export const getLocalFiles = async () => {
2021-05-29 06:27:52 +00:00
const files: Array<File> = (await localForage.getItem<File[]>(FILES)) || [];
return files;
};
2021-08-13 02:38:38 +00:00
export const syncFiles = async (
collections: Collection[],
setFiles: (files: File[]) => void
) => {
const localFiles = await getLocalFiles();
let files = await removeDeletedCollectionFiles(collections, localFiles);
2021-05-29 06:27:52 +00:00
if (files.length !== localFiles.length) {
2021-04-24 13:30:11 +00:00
await localForage.setItem('files', files);
setFiles(files);
}
2021-05-29 06:27:52 +00:00
for (const collection of collections) {
if (!getToken()) {
continue;
}
2021-08-13 02:38:38 +00:00
const lastSyncTime =
(await localForage.getItem<number>(`${collection.id}-time`)) ?? 0;
if (collection.updationTime === lastSyncTime) {
2021-02-08 07:33:46 +00:00
continue;
2021-02-08 11:01:05 +00:00
}
2021-08-13 02:38:38 +00:00
const fetchedFiles =
(await getFiles(
collection,
lastSyncTime,
DIFF_LIMIT,
files,
setFiles
)) ?? [];
2021-02-08 10:56:47 +00:00
files.push(...fetchedFiles);
2021-05-29 06:27:52 +00:00
const latestVersionFiles = new Map<string, File>();
2021-02-08 10:56:47 +00:00
files.forEach((file) => {
2021-03-16 09:41:11 +00:00
const uid = `${file.collectionID}-${file.id}`;
if (
2021-03-16 09:41:11 +00:00
!latestVersionFiles.has(uid) ||
latestVersionFiles.get(uid).updationTime < file.updationTime
) {
2021-03-16 09:41:11 +00:00
latestVersionFiles.set(uid, file);
2021-02-08 10:56:47 +00:00
}
});
files = [];
2021-05-29 06:27:52 +00:00
// eslint-disable-next-line @typescript-eslint/no-unused-vars
2021-02-08 10:56:47 +00:00
for (const [_, file] of latestVersionFiles) {
if (file.isDeleted) {
2021-02-08 10:56:47 +00:00
continue;
}
files.push(file);
2021-02-06 07:54:36 +00:00
}
2021-08-13 02:38:38 +00:00
files = sortFiles(files);
2021-02-08 10:56:47 +00:00
await localForage.setItem('files', files);
await localForage.setItem(
`${collection.id}-time`,
2021-08-13 02:38:38 +00:00
collection.updationTime
);
setFiles(
files.map((item) => ({
...item,
w: window.innerWidth,
h: window.innerHeight,
}))
);
}
return {
files: files.map((item) => ({
...item,
w: window.innerWidth,
h: window.innerHeight,
})),
};
};
export const getFiles = async (
collection: Collection,
2021-02-09 05:07:46 +00:00
sinceTime: number,
2021-05-29 06:27:52 +00:00
limit: number,
2021-06-01 20:42:44 +00:00
files: File[],
2021-08-13 02:38:38 +00:00
setFiles: (files: File[]) => void
): Promise<File[]> => {
2021-01-25 07:05:45 +00:00
try {
const decryptedFiles: File[] = [];
2021-08-13 02:38:38 +00:00
let time =
sinceTime ||
2021-02-09 05:07:46 +00:00
(await localForage.getItem<number>(`${collection.id}-time`)) ||
0;
let resp;
do {
const token = getToken();
if (!token) {
break;
}
resp = await HTTPService.get(
`${ENDPOINT}/collections/diff`,
{
2021-02-17 08:35:19 +00:00
collectionID: collection.id,
sinceTime: time,
2021-05-29 06:27:52 +00:00
limit,
},
{
'X-Auth-Token': token,
2021-08-13 02:38:38 +00:00
}
);
decryptedFiles.push(
...(await Promise.all(
resp.data.diff.map(async (file: File) => {
if (!file.isDeleted) {
2021-07-25 07:56:05 +00:00
file = await decryptFile(file, collection);
}
return file;
2021-08-13 02:38:38 +00:00
}) as Promise<File>[]
))
);
if (resp.data.diff.length) {
2021-02-17 08:35:19 +00:00
time = resp.data.diff.slice(-1)[0].updationTime;
}
2021-08-13 02:38:38 +00:00
setFiles(
[...(files || []), ...decryptedFiles]
.filter((item) => !item.isDeleted)
.sort(
(a, b) =>
b.metadata.creationTime - a.metadata.creationTime
)
);
} while (resp.data.diff.length === limit);
return decryptedFiles;
2021-01-25 07:05:45 +00:00
} catch (e) {
2021-06-12 17:14:21 +00:00
logError(e, 'Get files failed');
}
};
const removeDeletedCollectionFiles = async (
collections: Collection[],
2021-08-13 02:38:38 +00:00
files: File[]
) => {
const syncedCollectionIds = new Set<number>();
2021-05-29 06:27:52 +00:00
for (const collection of collections) {
syncedCollectionIds.add(collection.id);
}
files = files.filter((file) => syncedCollectionIds.has(file.collectionID));
2021-02-09 05:07:46 +00:00
return files;
};
2021-03-21 06:53:41 +00:00
export const deleteFiles = async (filesToDelete: number[]) => {
2021-03-21 06:53:41 +00:00
try {
const token = getToken();
if (!token) {
2021-03-30 05:09:43 +00:00
return;
2021-03-21 06:53:41 +00:00
}
await HTTPService.post(
`${ENDPOINT}/files/delete`,
2021-05-30 16:56:48 +00:00
{ fileIDs: filesToDelete },
2021-03-21 06:53:41 +00:00
null,
{
'X-Auth-Token': token,
2021-08-13 02:38:38 +00:00
}
2021-03-21 06:53:41 +00:00
);
} catch (e) {
2021-06-12 17:14:21 +00:00
logError(e, 'delete failed');
2021-05-31 10:58:33 +00:00
throw e;
2021-03-21 06:53:41 +00:00
}
};
2021-09-21 08:03:15 +00:00
2021-09-21 10:37:53 +00:00
export const updateMagicMetadata = async (files: File[]) => {
2021-09-21 08:03:15 +00:00
const token = getToken();
if (!token) {
return;
}
2021-09-21 10:37:53 +00:00
const reqBody: UpdateMagicMetadataRequest = { metadataList: [] };
for (const file of files) {
reqBody.metadataList.push({
id: file.id,
magicMetadata: file.magicMetadata,
});
}
await HTTPService.put(`${ENDPOINT}/files/magic-metadata`, reqBody, null, {
'X-Auth-Token': token,
});
2021-09-21 08:03:15 +00:00
};