signal-desktop/ts/util/uploads/uploads.ts
Jamie Kyle 8ef0ec706d
Add utilities for using TUS Protocol
Co-authored-by: Scott Nonnenberg <scott@signal.org>
Co-authored-by: Fedor Indutny <indutny@signal.org>
2024-04-30 17:57:57 -07:00

136 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,
});
}