2021-01-06 17:23:15 +00:00
|
|
|
// Copyright 2018-2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-05-25 22:40:04 +00:00
|
|
|
import { z } from 'zod';
|
2021-01-06 17:23:15 +00:00
|
|
|
import FormData from 'form-data';
|
|
|
|
import { gzip } from 'zlib';
|
|
|
|
import pify from 'pify';
|
2021-02-19 21:17:10 +00:00
|
|
|
import got, { Response } from 'got';
|
2021-01-06 17:23:15 +00:00
|
|
|
import { getUserAgent } from '../util/getUserAgent';
|
2021-05-13 17:18:51 +00:00
|
|
|
import { maybeParseUrl } from '../util/url';
|
2021-01-06 17:23:15 +00:00
|
|
|
|
|
|
|
const BASE_URL = 'https://debuglogs.org';
|
|
|
|
|
2021-04-01 17:37:10 +00:00
|
|
|
const tokenBodySchema = z
|
|
|
|
.object({
|
|
|
|
fields: z.record(z.unknown()),
|
|
|
|
url: z.string(),
|
|
|
|
})
|
|
|
|
.nonstrict();
|
2021-01-06 17:23:15 +00:00
|
|
|
|
|
|
|
const parseTokenBody = (
|
2021-04-01 17:37:10 +00:00
|
|
|
rawBody: unknown
|
2021-01-06 17:23:15 +00:00
|
|
|
): { fields: Record<string, unknown>; url: string } => {
|
2021-04-01 17:37:10 +00:00
|
|
|
const body = tokenBodySchema.parse(rawBody);
|
2021-01-06 17:23:15 +00:00
|
|
|
|
2021-05-13 17:18:51 +00:00
|
|
|
const parsedUrl = maybeParseUrl(body.url);
|
|
|
|
if (!parsedUrl) {
|
2021-01-06 17:23:15 +00:00
|
|
|
throw new Error("Token body's URL was not a valid URL");
|
|
|
|
}
|
|
|
|
if (parsedUrl.protocol !== 'https:') {
|
|
|
|
throw new Error("Token body's URL was not HTTPS");
|
|
|
|
}
|
|
|
|
|
2021-04-01 17:37:10 +00:00
|
|
|
return body;
|
2021-01-06 17:23:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
export const uploadDebugLogs = async (
|
|
|
|
content: string,
|
|
|
|
appVersion: string
|
|
|
|
): Promise<string> => {
|
|
|
|
const headers = { 'User-Agent': getUserAgent(appVersion) };
|
|
|
|
|
|
|
|
const signedForm = await got.get(BASE_URL, { json: true, headers });
|
|
|
|
const { fields, url } = parseTokenBody(signedForm.body);
|
|
|
|
|
|
|
|
const form = new FormData();
|
|
|
|
// The API expects `key` to be the first field:
|
|
|
|
form.append('key', fields.key);
|
|
|
|
Object.entries(fields)
|
|
|
|
.filter(([key]) => key !== 'key')
|
|
|
|
.forEach(([key, value]) => {
|
|
|
|
form.append(key, value);
|
|
|
|
});
|
|
|
|
|
|
|
|
const contentBuffer = await pify(gzip)(Buffer.from(content, 'utf8'));
|
|
|
|
const contentType = 'application/gzip';
|
|
|
|
form.append('Content-Type', contentType);
|
|
|
|
form.append('file', contentBuffer, {
|
|
|
|
contentType,
|
|
|
|
filename: `signal-desktop-debug-log-${appVersion}.txt.gz`,
|
|
|
|
});
|
|
|
|
|
|
|
|
window.log.info('Debug log upload starting...');
|
2021-02-19 21:17:10 +00:00
|
|
|
try {
|
|
|
|
const { statusCode, body } = await got.post(url, { headers, body: form });
|
|
|
|
if (statusCode !== 204) {
|
|
|
|
throw new Error(
|
|
|
|
`Failed to upload to S3, got status ${statusCode}, body '${body}'`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
const response = error.response as Response<string>;
|
|
|
|
throw new Error(
|
|
|
|
`Got threw on upload to S3, got status ${response?.statusCode}, body '${response?.body}' `
|
|
|
|
);
|
2021-01-06 17:23:15 +00:00
|
|
|
}
|
|
|
|
window.log.info('Debug log upload complete.');
|
|
|
|
|
|
|
|
return `${BASE_URL}/${fields.key}`;
|
|
|
|
};
|