137 lines
2.9 KiB
TypeScript
137 lines
2.9 KiB
TypeScript
|
// Copyright 2024 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
import { createReadStream, createWriteStream } from 'node:fs';
|
||
|
import { Writable } from 'node:stream';
|
||
|
import type { TusFileReader } from './tusProtocol';
|
||
|
import { tusResumeUpload, tusUpload } from './tusProtocol';
|
||
|
import { HTTPError } from '../../textsecure/Errors';
|
||
|
|
||
|
const defaultFileReader: TusFileReader = (filePath, offset) => {
|
||
|
return createReadStream(filePath, { start: offset });
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @public
|
||
|
* Uploads a file to the attachments bucket.
|
||
|
* @throws {ResponseError} If the server responded with an error.
|
||
|
*/
|
||
|
export async function uploadAttachment({
|
||
|
host,
|
||
|
fileName,
|
||
|
filePath,
|
||
|
fileSize,
|
||
|
checksum,
|
||
|
headers = {},
|
||
|
signal,
|
||
|
}: {
|
||
|
host: string;
|
||
|
fileName: string;
|
||
|
filePath: string;
|
||
|
fileSize: number;
|
||
|
checksum: string;
|
||
|
headers?: Record<string, string>;
|
||
|
signal?: AbortSignal;
|
||
|
}): Promise<void> {
|
||
|
return tusUpload({
|
||
|
endpoint: `${host}/upload/attachments`,
|
||
|
headers: {
|
||
|
...headers,
|
||
|
'X-Signal-Checksum-Sha256': checksum,
|
||
|
},
|
||
|
fileName,
|
||
|
filePath,
|
||
|
fileSize,
|
||
|
reader: defaultFileReader,
|
||
|
signal,
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @public
|
||
|
* Resumes an upload to the attachments bucket.
|
||
|
* @throws {ResponseError} If the server responded with an error.
|
||
|
*/
|
||
|
export async function resumeUploadAttachment({
|
||
|
host,
|
||
|
fileName,
|
||
|
filePath,
|
||
|
fileSize,
|
||
|
headers = {},
|
||
|
signal,
|
||
|
}: {
|
||
|
host: string;
|
||
|
fileName: string;
|
||
|
filePath: string;
|
||
|
fileSize: number;
|
||
|
headers?: Record<string, string>;
|
||
|
signal?: AbortSignal;
|
||
|
}): Promise<void> {
|
||
|
return tusResumeUpload({
|
||
|
endpoint: `${host}/upload/attachments`,
|
||
|
headers,
|
||
|
fileName,
|
||
|
filePath,
|
||
|
fileSize,
|
||
|
reader: defaultFileReader,
|
||
|
signal,
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Downloads a file with Signal headers.
|
||
|
* @throws {ResponseError} If the server responded with an error.
|
||
|
* @throws {Error} If the response has no body.
|
||
|
*/
|
||
|
export async function _doDownload({
|
||
|
endpoint,
|
||
|
headers = {},
|
||
|
filePath,
|
||
|
signal,
|
||
|
}: {
|
||
|
endpoint: string;
|
||
|
filePath: string;
|
||
|
headers?: Record<string, string>;
|
||
|
signal?: AbortSignal;
|
||
|
}): Promise<void> {
|
||
|
const response = await fetch(endpoint, {
|
||
|
method: 'GET',
|
||
|
signal,
|
||
|
redirect: 'error',
|
||
|
headers,
|
||
|
});
|
||
|
if (!response.ok) {
|
||
|
throw HTTPError.fromResponse(response);
|
||
|
}
|
||
|
if (!response.body) {
|
||
|
throw new Error('Response has no body');
|
||
|
}
|
||
|
const writable = createWriteStream(filePath);
|
||
|
await response.body.pipeTo(Writable.toWeb(writable));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @public
|
||
|
* Downloads a file from the attachments bucket.
|
||
|
* @throws {ResponseError} If the server responded with an error.
|
||
|
*/
|
||
|
export async function downloadAttachment({
|
||
|
host,
|
||
|
fileName,
|
||
|
filePath,
|
||
|
headers,
|
||
|
signal,
|
||
|
}: {
|
||
|
host: string;
|
||
|
fileName: string;
|
||
|
filePath: string;
|
||
|
headers?: Record<string, string>;
|
||
|
signal?: AbortSignal;
|
||
|
}): Promise<void> {
|
||
|
return _doDownload({
|
||
|
endpoint: `${host}/attachments/${fileName}`,
|
||
|
headers,
|
||
|
filePath,
|
||
|
signal,
|
||
|
});
|
||
|
}
|