import crypto from 'crypto'; import path from 'path'; import pify from 'pify'; import { default as glob } from 'glob'; import fse from 'fs-extra'; import toArrayBuffer from 'to-arraybuffer'; import { isArrayBuffer, isString, map } from 'lodash'; import { decryptAttachmentBuffer, encryptAttachmentBuffer } from '../../ts/types/Attachment'; const PATH = 'attachments.noindex'; export const getAllAttachments = async (userDataPath: string) => { const dir = exports.getPath(userDataPath); const pattern = path.join(dir, '**', '*'); const files = await pify(glob)(pattern, { nodir: true }); return map(files, file => path.relative(dir, file)); }; // getPath :: AbsolutePath -> AbsolutePath export const getPath = (userDataPath: string) => { if (!isString(userDataPath)) { throw new TypeError("'userDataPath' must be a string"); } return path.join(userDataPath, PATH); }; // ensureDirectory :: AbsolutePath -> IO Unit export const ensureDirectory = async (userDataPath: string) => { if (!isString(userDataPath)) { throw new TypeError("'userDataPath' must be a string"); } await fse.ensureDir(exports.getPath(userDataPath)); }; // createReader :: AttachmentsPath -> // RelativePath -> // IO (Promise ArrayBuffer) export const createReader = (root: string) => { if (!isString(root)) { throw new TypeError("'root' must be a path"); } return async (relativePath: string) => { if (!isString(relativePath)) { throw new TypeError("'relativePath' must be a string"); } const absolutePath = path.join(root, relativePath); const normalized = path.normalize(absolutePath); if (!normalized.startsWith(root)) { throw new Error('Invalid relative path'); } const buffer = await fse.readFile(normalized); const decryptedData = await decryptAttachmentBuffer(toArrayBuffer(buffer)); return decryptedData.buffer; }; }; // createWriterForNew :: AttachmentsPath -> // ArrayBuffer -> // IO (Promise RelativePath) export const createWriterForNew = (root: string) => { if (!isString(root)) { throw new TypeError("'root' must be a path"); } return async (arrayBuffer: ArrayBuffer) => { if (!isArrayBuffer(arrayBuffer)) { throw new TypeError("'arrayBuffer' must be an array buffer"); } const name = exports.createName(); const relativePath = exports.getRelativePath(name); return exports.createWriterForExisting(root)({ data: arrayBuffer, path: relativePath, }); }; }; // createWriter :: AttachmentsPath -> // { data: ArrayBuffer, path: RelativePath } -> // IO (Promise RelativePath) export const createWriterForExisting = (root: any) => { if (!isString(root)) { throw new TypeError("'root' must be a path"); } return async ({ data: arrayBuffer, path: relativePath, }: { data?: ArrayBuffer; path?: string } = {}) => { if (!isString(relativePath)) { throw new TypeError("'relativePath' must be a path"); } if (!isArrayBuffer(arrayBuffer)) { throw new TypeError("'arrayBuffer' must be an array buffer"); } const absolutePath = path.join(root, relativePath); const normalized = path.normalize(absolutePath); if (!normalized.startsWith(root)) { throw new Error('Invalid relative path'); } await fse.ensureFile(normalized); const { encryptedBufferWithHeader } = await encryptAttachmentBuffer(arrayBuffer); const buffer = Buffer.from(encryptedBufferWithHeader.buffer); await fse.writeFile(normalized, buffer); return relativePath; }; }; // createDeleter :: AttachmentsPath -> // RelativePath -> // IO Unit export const createDeleter = (root: any) => { if (!isString(root)) { throw new TypeError("'root' must be a path"); } return async (relativePath: any) => { if (!isString(relativePath)) { throw new TypeError("'relativePath' must be a string"); } const absolutePath = path.join(root, relativePath); const normalized = path.normalize(absolutePath); if (!normalized.startsWith(root)) { throw new Error('Invalid relative path'); } await fse.remove(absolutePath); }; }; export const deleteAll = async ({ userDataPath, attachments }: any) => { const deleteFromDisk = exports.createDeleter(exports.getPath(userDataPath)); // tslint:disable-next-line: one-variable-per-declaration for (let index = 0, max = attachments.length; index < max; index += 1) { const file = attachments[index]; // eslint-disable-next-line no-await-in-loop await deleteFromDisk(file); } // tslint:disable-next-line: no-console console.log(`deleteAll: deleted ${attachments.length} files`); }; // createName :: Unit -> IO String export const createName = () => { const buffer = crypto.randomBytes(32); return buffer.toString('hex'); }; // getRelativePath :: String -> Path export const getRelativePath = (name: any) => { if (!isString(name)) { throw new TypeError("'name' must be a string"); } const prefix = name.slice(0, 2); return path.join(prefix, name); }; // createAbsolutePathGetter :: RootPath -> RelativePath -> AbsolutePath export const createAbsolutePathGetter = (rootPath: string) => (relativePath: string) => { const absolutePath = path.join(rootPath, relativePath); const normalized = path.normalize(absolutePath); if (!normalized.startsWith(rootPath)) { throw new Error('Invalid relative path'); } return normalized; };