2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2018 Signal Messenger, LLC
|
2021-10-27 17:54:16 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { ipcRenderer } from 'electron';
|
|
|
|
import { isString, isTypedArray } from 'lodash';
|
|
|
|
import { join, normalize, basename } from 'path';
|
|
|
|
import fse from 'fs-extra';
|
2024-10-08 03:17:03 +00:00
|
|
|
import { v4 as getGuid } from 'uuid';
|
2021-10-27 17:54:16 +00:00
|
|
|
|
|
|
|
import { isPathInside } from '../util/isPathInside';
|
|
|
|
import { writeWindowsZoneIdentifier } from '../util/windowsZoneIdentifier';
|
2023-04-20 21:23:19 +00:00
|
|
|
import OS from '../util/os/osMain';
|
2024-07-11 19:44:09 +00:00
|
|
|
import { getRelativePath, createName } from '../util/attachmentPath';
|
2021-10-27 17:54:16 +00:00
|
|
|
|
2024-10-08 21:45:00 +00:00
|
|
|
export * from '../util/ensureAttachmentIsReencryptable';
|
2022-10-24 20:46:36 +00:00
|
|
|
export * from '../../app/attachments';
|
2021-10-27 17:54:16 +00:00
|
|
|
|
|
|
|
type FSAttrType = {
|
|
|
|
set: (path: string, attribute: string, value: string) => Promise<void>;
|
|
|
|
};
|
|
|
|
|
|
|
|
let xattr: FSAttrType | undefined;
|
|
|
|
|
|
|
|
try {
|
2022-06-03 21:07:51 +00:00
|
|
|
// eslint-disable-next-line global-require, import/no-extraneous-dependencies
|
2021-10-27 17:54:16 +00:00
|
|
|
xattr = require('fs-xattr');
|
|
|
|
} catch (e) {
|
2023-02-22 04:21:30 +00:00
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
throw e;
|
|
|
|
}
|
2021-10-27 17:54:16 +00:00
|
|
|
window.SignalContext.log?.info('x-attr dependency did not load successfully');
|
|
|
|
}
|
|
|
|
|
2024-07-11 19:44:09 +00:00
|
|
|
export const createPlaintextReader = (
|
2021-10-27 17:54:16 +00:00
|
|
|
root: string
|
|
|
|
): ((relativePath: string) => Promise<Uint8Array>) => {
|
|
|
|
if (!isString(root)) {
|
|
|
|
throw new TypeError("'root' must be a path");
|
|
|
|
}
|
|
|
|
|
|
|
|
return async (relativePath: string): Promise<Uint8Array> => {
|
|
|
|
if (!isString(relativePath)) {
|
|
|
|
throw new TypeError("'relativePath' must be a string");
|
|
|
|
}
|
|
|
|
|
|
|
|
const absolutePath = join(root, relativePath);
|
|
|
|
const normalized = normalize(absolutePath);
|
|
|
|
if (!isPathInside(normalized, root)) {
|
|
|
|
throw new Error('Invalid relative path');
|
|
|
|
}
|
|
|
|
return fse.readFile(normalized);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export const copyIntoAttachmentsDirectory = (
|
|
|
|
root: string
|
|
|
|
): ((sourcePath: string) => Promise<{ path: string; size: number }>) => {
|
|
|
|
if (!isString(root)) {
|
|
|
|
throw new TypeError("'root' must be a path");
|
|
|
|
}
|
|
|
|
|
|
|
|
const userDataPath = window.SignalContext.getPath('userData');
|
|
|
|
|
|
|
|
return async (
|
|
|
|
sourcePath: string
|
|
|
|
): Promise<{ path: string; size: number }> => {
|
|
|
|
if (!isString(sourcePath)) {
|
|
|
|
throw new TypeError('sourcePath must be a string');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isPathInside(sourcePath, userDataPath)) {
|
|
|
|
throw new Error(
|
|
|
|
"'sourcePath' must be relative to the user config directory"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const name = createName();
|
|
|
|
const relativePath = getRelativePath(name);
|
|
|
|
const absolutePath = join(root, relativePath);
|
|
|
|
const normalized = normalize(absolutePath);
|
|
|
|
if (!isPathInside(normalized, root)) {
|
|
|
|
throw new Error('Invalid relative path');
|
|
|
|
}
|
|
|
|
|
|
|
|
await fse.ensureFile(normalized);
|
|
|
|
await fse.copy(sourcePath, normalized);
|
|
|
|
const { size } = await fse.stat(normalized);
|
|
|
|
|
|
|
|
return {
|
|
|
|
path: relativePath,
|
|
|
|
size,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export const createWriterForNew = (
|
2021-11-02 23:01:13 +00:00
|
|
|
root: string,
|
|
|
|
suffix?: string
|
2021-10-27 17:54:16 +00:00
|
|
|
): ((bytes: Uint8Array) => Promise<string>) => {
|
|
|
|
if (!isString(root)) {
|
|
|
|
throw new TypeError("'root' must be a path");
|
|
|
|
}
|
|
|
|
|
|
|
|
return async (bytes: Uint8Array) => {
|
|
|
|
if (!isTypedArray(bytes)) {
|
|
|
|
throw new TypeError("'bytes' must be a typed array");
|
|
|
|
}
|
|
|
|
|
2021-11-02 23:01:13 +00:00
|
|
|
const name = createName(suffix);
|
2021-10-27 17:54:16 +00:00
|
|
|
const relativePath = getRelativePath(name);
|
|
|
|
return createWriterForExisting(root)({
|
|
|
|
data: bytes,
|
|
|
|
path: relativePath,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-07-11 19:44:09 +00:00
|
|
|
const createWriterForExisting = (
|
2021-10-27 17:54:16 +00:00
|
|
|
root: string
|
2022-06-13 21:39:35 +00:00
|
|
|
): ((options: { data?: Uint8Array; path?: string }) => Promise<string>) => {
|
2021-10-27 17:54:16 +00:00
|
|
|
if (!isString(root)) {
|
|
|
|
throw new TypeError("'root' must be a path");
|
|
|
|
}
|
|
|
|
|
|
|
|
return async ({
|
|
|
|
data: bytes,
|
|
|
|
path: relativePath,
|
|
|
|
}: {
|
2022-06-13 21:39:35 +00:00
|
|
|
data?: Uint8Array;
|
|
|
|
path?: string;
|
2021-10-27 17:54:16 +00:00
|
|
|
}): Promise<string> => {
|
|
|
|
if (!isString(relativePath)) {
|
|
|
|
throw new TypeError("'relativePath' must be a path");
|
|
|
|
}
|
|
|
|
|
2022-06-13 21:39:35 +00:00
|
|
|
if (!bytes) {
|
|
|
|
throw new TypeError("'data' must be a Uint8Array");
|
2021-10-27 17:54:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const buffer = Buffer.from(bytes);
|
|
|
|
const absolutePath = join(root, relativePath);
|
|
|
|
const normalized = normalize(absolutePath);
|
|
|
|
if (!isPathInside(normalized, root)) {
|
|
|
|
throw new Error('Invalid relative path');
|
|
|
|
}
|
|
|
|
|
|
|
|
await fse.ensureFile(normalized);
|
|
|
|
await fse.writeFile(normalized, buffer);
|
|
|
|
return relativePath;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-11-11 22:43:05 +00:00
|
|
|
export const createAbsolutePathGetter =
|
|
|
|
(rootPath: string) =>
|
|
|
|
(relativePath: string): string => {
|
|
|
|
const absolutePath = join(rootPath, relativePath);
|
|
|
|
const normalized = normalize(absolutePath);
|
|
|
|
if (!isPathInside(normalized, rootPath)) {
|
|
|
|
throw new Error('Invalid relative path');
|
|
|
|
}
|
|
|
|
return normalized;
|
|
|
|
};
|
2021-10-27 17:54:16 +00:00
|
|
|
|
|
|
|
export const createDoesExist = (
|
|
|
|
root: string
|
|
|
|
): ((relativePath: string) => Promise<boolean>) => {
|
|
|
|
if (!isString(root)) {
|
|
|
|
throw new TypeError("'root' must be a path");
|
|
|
|
}
|
|
|
|
|
|
|
|
return async (relativePath: string): Promise<boolean> => {
|
|
|
|
if (!isString(relativePath)) {
|
|
|
|
throw new TypeError("'relativePath' must be a string");
|
|
|
|
}
|
|
|
|
|
|
|
|
const absolutePath = join(root, relativePath);
|
|
|
|
const normalized = normalize(absolutePath);
|
|
|
|
if (!isPathInside(normalized, root)) {
|
|
|
|
throw new Error('Invalid relative path');
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
await fse.access(normalized, fse.constants.F_OK);
|
|
|
|
return true;
|
|
|
|
} catch (error) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
const showSaveDialog = (
|
|
|
|
defaultPath: string
|
|
|
|
): Promise<{
|
|
|
|
canceled: boolean;
|
|
|
|
filePath?: string;
|
|
|
|
}> => {
|
|
|
|
return ipcRenderer.invoke('show-save-dialog', { defaultPath });
|
|
|
|
};
|
|
|
|
|
|
|
|
async function writeWithAttributes(
|
|
|
|
target: string,
|
|
|
|
data: Uint8Array
|
|
|
|
): Promise<void> {
|
|
|
|
await fse.writeFile(target, Buffer.from(data));
|
|
|
|
|
|
|
|
if (process.platform === 'darwin' && xattr) {
|
|
|
|
// kLSQuarantineTypeInstantMessageAttachment
|
|
|
|
const type = '0003';
|
|
|
|
|
|
|
|
// Hexadecimal seconds since epoch
|
|
|
|
const timestamp = Math.trunc(Date.now() / 1000).toString(16);
|
|
|
|
|
|
|
|
const appName = 'Signal';
|
|
|
|
const guid = getGuid();
|
|
|
|
|
|
|
|
// https://ilostmynotes.blogspot.com/2012/06/gatekeeper-xprotect-and-quarantine.html
|
|
|
|
const attrValue = `${type};${timestamp};${appName};${guid}`;
|
|
|
|
|
|
|
|
await xattr.set(target, 'com.apple.quarantine', attrValue);
|
2023-04-20 21:23:19 +00:00
|
|
|
} else if (OS.isWindows()) {
|
2021-10-27 17:54:16 +00:00
|
|
|
// This operation may fail (see the function's comments), which is not a show-stopper.
|
|
|
|
try {
|
|
|
|
await writeWindowsZoneIdentifier(target);
|
|
|
|
} catch (err) {
|
|
|
|
window.SignalContext.log?.warn(
|
|
|
|
'Failed to write Windows Zone.Identifier file; continuing'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const saveAttachmentToDisk = async ({
|
|
|
|
data,
|
|
|
|
name,
|
2024-10-23 21:44:12 +00:00
|
|
|
baseDir,
|
2021-10-27 17:54:16 +00:00
|
|
|
}: {
|
|
|
|
data: Uint8Array;
|
|
|
|
name: string;
|
2024-10-23 21:44:12 +00:00
|
|
|
/**
|
|
|
|
* Base directory for saving the attachment.
|
|
|
|
* If omitted, a dialog will be opened to let the user choose a directory
|
|
|
|
*/
|
|
|
|
baseDir?: string;
|
2021-10-27 17:54:16 +00:00
|
|
|
}): Promise<null | { fullPath: string; name: string }> => {
|
2024-10-23 21:44:12 +00:00
|
|
|
let filePath;
|
2021-10-27 17:54:16 +00:00
|
|
|
|
2024-10-23 21:44:12 +00:00
|
|
|
if (!baseDir) {
|
|
|
|
const { canceled, filePath: dialogFilePath } = await showSaveDialog(name);
|
|
|
|
if (canceled) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (!dialogFilePath) {
|
|
|
|
throw new Error(
|
|
|
|
"saveAttachmentToDisk: Dialog wasn't canceled, but returned path to attachment is null!"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
filePath = dialogFilePath;
|
|
|
|
} else {
|
|
|
|
filePath = join(baseDir, name);
|
2021-10-27 17:54:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
await writeWithAttributes(filePath, data);
|
|
|
|
|
|
|
|
const fileBasename = basename(filePath);
|
|
|
|
|
|
|
|
return {
|
|
|
|
fullPath: filePath,
|
|
|
|
name: fileBasename,
|
|
|
|
};
|
|
|
|
};
|