2018-03-16 21:32:17 +00:00
|
|
|
const crypto = require('crypto');
|
|
|
|
const path = require('path');
|
2018-04-02 19:08:53 +00:00
|
|
|
|
|
|
|
const fse = require('fs-extra');
|
2018-03-16 21:32:17 +00:00
|
|
|
const toArrayBuffer = require('to-arraybuffer');
|
2018-04-02 19:08:53 +00:00
|
|
|
const { isArrayBuffer, isString } = require('lodash');
|
2018-03-16 21:32:17 +00:00
|
|
|
|
|
|
|
|
2018-03-20 22:03:22 +00:00
|
|
|
const PATH = 'attachments.noindex';
|
2018-03-16 21:32:17 +00:00
|
|
|
|
|
|
|
// getPath :: AbsolutePath -> AbsolutePath
|
|
|
|
exports.getPath = (userDataPath) => {
|
|
|
|
if (!isString(userDataPath)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'userDataPath' must be a string");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
return path.join(userDataPath, PATH);
|
|
|
|
};
|
|
|
|
|
|
|
|
// ensureDirectory :: AbsolutePath -> IO Unit
|
|
|
|
exports.ensureDirectory = async (userDataPath) => {
|
|
|
|
if (!isString(userDataPath)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'userDataPath' must be a string");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
await fse.ensureDir(exports.getPath(userDataPath));
|
|
|
|
};
|
|
|
|
|
2018-04-03 19:25:24 +00:00
|
|
|
// createReader :: AttachmentsPath ->
|
|
|
|
// RelativePath ->
|
|
|
|
// IO (Promise ArrayBuffer)
|
|
|
|
exports.createReader = (root) => {
|
2018-03-16 21:32:17 +00:00
|
|
|
if (!isString(root)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'root' must be a path");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return async (relativePath) => {
|
|
|
|
if (!isString(relativePath)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'relativePath' must be a string");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const absolutePath = path.join(root, relativePath);
|
|
|
|
const buffer = await fse.readFile(absolutePath);
|
|
|
|
return toArrayBuffer(buffer);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2018-04-04 01:06:29 +00:00
|
|
|
// createWriterForNew :: AttachmentsPath ->
|
|
|
|
// ArrayBuffer ->
|
|
|
|
// IO (Promise RelativePath)
|
|
|
|
exports.createWriterForNew = (root) => {
|
2018-03-16 21:32:17 +00:00
|
|
|
if (!isString(root)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'root' must be a path");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return async (arrayBuffer) => {
|
|
|
|
if (!isArrayBuffer(arrayBuffer)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'arrayBuffer' must be an array buffer");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const name = exports.createName();
|
|
|
|
const relativePath = exports.getRelativePath(name);
|
2018-04-04 01:08:43 +00:00
|
|
|
return exports.createWriterForExisting(root)({
|
|
|
|
data: arrayBuffer,
|
|
|
|
path: relativePath,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
// createWriter :: AttachmentsPath ->
|
|
|
|
// { data: ArrayBuffer, path: RelativePath } ->
|
|
|
|
// IO (Promise RelativePath)
|
|
|
|
exports.createWriterForExisting = (root) => {
|
|
|
|
if (!isString(root)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'root' must be a path");
|
2018-04-04 01:08:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return async ({ data: arrayBuffer, path: relativePath } = {}) => {
|
|
|
|
if (!isString(relativePath)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'relativePath' must be a path");
|
2018-04-04 01:08:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!isArrayBuffer(arrayBuffer)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'arrayBuffer' must be an array buffer");
|
2018-04-04 01:08:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const buffer = Buffer.from(arrayBuffer);
|
2018-03-16 21:32:17 +00:00
|
|
|
const absolutePath = path.join(root, relativePath);
|
|
|
|
await fse.ensureFile(absolutePath);
|
|
|
|
await fse.writeFile(absolutePath, buffer);
|
|
|
|
return relativePath;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2018-04-03 19:25:24 +00:00
|
|
|
// createDeleter :: AttachmentsPath ->
|
|
|
|
// RelativePath ->
|
|
|
|
// IO Unit
|
|
|
|
exports.createDeleter = (root) => {
|
2018-03-19 23:50:14 +00:00
|
|
|
if (!isString(root)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'root' must be a path");
|
2018-03-19 23:50:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return async (relativePath) => {
|
|
|
|
if (!isString(relativePath)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'relativePath' must be a string");
|
2018-03-19 23:50:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const absolutePath = path.join(root, relativePath);
|
|
|
|
await fse.remove(absolutePath);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2018-03-16 21:32:17 +00:00
|
|
|
// createName :: Unit -> IO String
|
|
|
|
exports.createName = () => {
|
|
|
|
const buffer = crypto.randomBytes(32);
|
|
|
|
return buffer.toString('hex');
|
|
|
|
};
|
|
|
|
|
|
|
|
// getRelativePath :: String -> IO Path
|
|
|
|
exports.getRelativePath = (name) => {
|
|
|
|
if (!isString(name)) {
|
2018-04-11 19:44:52 +00:00
|
|
|
throw new TypeError("'name' must be a string");
|
2018-03-16 21:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const prefix = name.slice(0, 2);
|
|
|
|
return path.join(prefix, name);
|
|
|
|
};
|