2018-04-13 03:34:16 +03:00
|
|
|
/*
|
|
|
|
Copyright 2017 OpenMarket Ltd
|
|
|
|
Copyright 2018 New Vector Ltd
|
2019-10-09 13:59:10 +03:00
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
2018-04-13 03:34:16 +03:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import pako from 'pako';
|
|
|
|
|
2019-12-21 00:13:46 +03:00
|
|
|
import {MatrixClientPeg} from '../MatrixClientPeg';
|
2018-04-13 03:34:16 +03:00
|
|
|
import PlatformPeg from '../PlatformPeg';
|
|
|
|
import { _t } from '../languageHandler';
|
2020-01-16 09:22:31 +03:00
|
|
|
import Tar from "tar-js";
|
2018-04-13 03:34:16 +03:00
|
|
|
|
2019-12-20 03:45:24 +03:00
|
|
|
import * as rageshake from './rageshake';
|
2018-04-13 03:34:16 +03:00
|
|
|
|
|
|
|
|
|
|
|
// polyfill textencoder if necessary
|
|
|
|
import * as TextEncodingUtf8 from 'text-encoding-utf-8';
|
2020-01-05 23:52:54 +03:00
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
2018-04-13 03:34:16 +03:00
|
|
|
let TextEncoder = window.TextEncoder;
|
|
|
|
if (!TextEncoder) {
|
|
|
|
TextEncoder = TextEncodingUtf8.TextEncoder;
|
|
|
|
}
|
|
|
|
|
2020-01-16 09:22:31 +03:00
|
|
|
async function collectBugReport(opts) {
|
2018-04-13 03:34:16 +03:00
|
|
|
opts = opts || {};
|
|
|
|
const progressCallback = opts.progressCallback || (() => {});
|
|
|
|
|
|
|
|
progressCallback(_t("Collecting app version information"));
|
|
|
|
let version = "UNKNOWN";
|
|
|
|
try {
|
|
|
|
version = await PlatformPeg.get().getAppVersion();
|
2018-10-12 06:05:59 +03:00
|
|
|
} catch (err) {} // PlatformPeg already logs this.
|
2018-04-13 03:34:16 +03:00
|
|
|
|
|
|
|
let userAgent = "UNKNOWN";
|
|
|
|
if (window.navigator && window.navigator.userAgent) {
|
|
|
|
userAgent = window.navigator.userAgent;
|
|
|
|
}
|
|
|
|
|
2020-02-14 17:58:37 +03:00
|
|
|
let installedPWA = "UNKNOWN";
|
|
|
|
try {
|
|
|
|
// Known to work at least for desktop Chrome
|
|
|
|
installedPWA = window.matchMedia('(display-mode: standalone)').matches;
|
|
|
|
} catch (e) { }
|
|
|
|
|
2020-02-14 20:36:14 +03:00
|
|
|
let touchInput = "UNKNOWN";
|
|
|
|
try {
|
|
|
|
// MDN claims broad support across browsers
|
|
|
|
touchInput = window.matchMedia('(pointer: coarse)').matches;
|
|
|
|
} catch (e) { }
|
|
|
|
|
2018-04-13 03:34:16 +03:00
|
|
|
const client = MatrixClientPeg.get();
|
|
|
|
|
|
|
|
console.log("Sending bug report.");
|
|
|
|
|
|
|
|
const body = new FormData();
|
|
|
|
body.append('text', opts.userText || "User did not supply any additional text.");
|
|
|
|
body.append('app', 'riot-web');
|
|
|
|
body.append('version', version);
|
|
|
|
body.append('user_agent', userAgent);
|
2020-02-14 17:58:37 +03:00
|
|
|
body.append('installed_pwa', installedPWA);
|
2020-02-14 20:36:14 +03:00
|
|
|
body.append('touch_input', touchInput);
|
2018-04-13 03:34:16 +03:00
|
|
|
|
|
|
|
if (client) {
|
|
|
|
body.append('user_id', client.credentials.userId);
|
|
|
|
body.append('device_id', client.deviceId);
|
|
|
|
}
|
|
|
|
|
2020-02-21 18:19:53 +03:00
|
|
|
const keys = [`ed25519:${client.getDeviceEd25519Key()}`];
|
|
|
|
if (client.getDeviceCurve25519Key) {
|
|
|
|
keys.push(`curve25519:${client.getDeviceCurve25519Key()}`);
|
|
|
|
}
|
|
|
|
body.append('device_keys', keys.join(', '));
|
|
|
|
body.append('cross_signing_key', client.getCrossSigningId());
|
|
|
|
|
2019-10-09 13:59:10 +03:00
|
|
|
if (opts.label) {
|
|
|
|
body.append('label', opts.label);
|
|
|
|
}
|
|
|
|
|
2020-01-05 23:52:54 +03:00
|
|
|
// add labs options
|
|
|
|
const enabledLabs = SettingsStore.getLabsFeatures().filter(SettingsStore.isFeatureEnabled);
|
|
|
|
if (enabledLabs.length) {
|
|
|
|
body.append('enabled_labs', enabledLabs.join(', '));
|
|
|
|
}
|
|
|
|
|
2020-02-20 03:38:08 +03:00
|
|
|
// add storage persistence/quota information
|
|
|
|
if (navigator.storage && navigator.storage.persisted) {
|
|
|
|
try {
|
|
|
|
body.append("storageManager_persisted", await navigator.storage.persisted());
|
|
|
|
} catch (e) {}
|
2020-03-25 14:07:11 +03:00
|
|
|
} else if (document.hasStorageAccess) { // Safari
|
2020-03-25 14:04:09 +03:00
|
|
|
try {
|
|
|
|
body.append("storageManager_persisted", await document.hasStorageAccess());
|
|
|
|
} catch (e) {}
|
|
|
|
}
|
2020-02-20 03:38:08 +03:00
|
|
|
if (navigator.storage && navigator.storage.estimate) {
|
|
|
|
try {
|
|
|
|
const estimate = await navigator.storage.estimate();
|
|
|
|
body.append("storageManager_quota", estimate.quota);
|
|
|
|
body.append("storageManager_usage", estimate.usage);
|
|
|
|
if (estimate.usageDetails) {
|
|
|
|
Object.keys(estimate.usageDetails).forEach(k => {
|
|
|
|
body.append(`storageManager_usage_${k}`, estimate.usageDetails[k]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (e) {}
|
|
|
|
}
|
|
|
|
|
2018-04-13 03:34:16 +03:00
|
|
|
if (opts.sendLogs) {
|
|
|
|
progressCallback(_t("Collecting logs"));
|
|
|
|
const logs = await rageshake.getLogsForReport();
|
2018-10-12 06:05:59 +03:00
|
|
|
for (const entry of logs) {
|
2018-04-13 03:34:16 +03:00
|
|
|
// encode as UTF-8
|
|
|
|
const buf = new TextEncoder().encode(entry.lines);
|
|
|
|
|
|
|
|
// compress
|
|
|
|
const compressed = pako.gzip(buf);
|
|
|
|
|
|
|
|
body.append('compressed-log', new Blob([compressed]), entry.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-16 09:22:31 +03:00
|
|
|
return body;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a bug report.
|
|
|
|
*
|
|
|
|
* @param {string} bugReportEndpoint HTTP url to send the report to
|
|
|
|
*
|
|
|
|
* @param {object} opts optional dictionary of options
|
|
|
|
*
|
|
|
|
* @param {string} opts.userText Any additional user input.
|
|
|
|
*
|
|
|
|
* @param {boolean} opts.sendLogs True to send logs
|
|
|
|
*
|
|
|
|
* @param {function(string)} opts.progressCallback Callback to call with progress updates
|
|
|
|
*
|
|
|
|
* @return {Promise} Resolved when the bug report is sent.
|
|
|
|
*/
|
|
|
|
export default async function sendBugReport(bugReportEndpoint, opts) {
|
|
|
|
if (!bugReportEndpoint) {
|
|
|
|
throw new Error("No bug report endpoint has been set.");
|
|
|
|
}
|
|
|
|
|
|
|
|
opts = opts || {};
|
|
|
|
const progressCallback = opts.progressCallback || (() => {});
|
|
|
|
const body = await collectBugReport(opts);
|
|
|
|
|
2018-04-13 03:34:16 +03:00
|
|
|
progressCallback(_t("Uploading report"));
|
|
|
|
await _submitReport(bugReportEndpoint, body, progressCallback);
|
|
|
|
}
|
|
|
|
|
2020-01-16 09:22:31 +03:00
|
|
|
/**
|
|
|
|
* Downloads the files from a bug report. This is the same as sendBugReport,
|
|
|
|
* but instead causes the browser to download the files locally.
|
|
|
|
*
|
|
|
|
* @param {object} opts optional dictionary of options
|
|
|
|
*
|
|
|
|
* @param {string} opts.userText Any additional user input.
|
|
|
|
*
|
|
|
|
* @param {boolean} opts.sendLogs True to send logs
|
|
|
|
*
|
|
|
|
* @param {function(string)} opts.progressCallback Callback to call with progress updates
|
|
|
|
*
|
|
|
|
* @return {Promise} Resolved when the bug report is downloaded (or started).
|
|
|
|
*/
|
|
|
|
export async function downloadBugReport(opts) {
|
|
|
|
opts = opts || {};
|
|
|
|
const progressCallback = opts.progressCallback || (() => {});
|
|
|
|
const body = await collectBugReport(opts);
|
|
|
|
|
|
|
|
progressCallback(_t("Downloading report"));
|
|
|
|
let metadata = "";
|
|
|
|
const tape = new Tar();
|
|
|
|
let i = 0;
|
|
|
|
for (const e of body.entries()) {
|
|
|
|
if (e[0] === 'compressed-log') {
|
|
|
|
await new Promise((resolve => {
|
|
|
|
const reader = new FileReader();
|
|
|
|
reader.addEventListener('loadend', ev => {
|
|
|
|
tape.append(`log-${i++}.log`, pako.ungzip(ev.target.result));
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
reader.readAsArrayBuffer(e[1]);
|
2020-03-30 18:12:28 +03:00
|
|
|
}));
|
2020-01-16 09:22:31 +03:00
|
|
|
} else {
|
|
|
|
metadata += `${e[0]} = ${e[1]}\n`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tape.append('issue.txt', metadata);
|
|
|
|
|
|
|
|
// We have to create a new anchor to download if we want a filename. Otherwise we could
|
|
|
|
// just use window.open.
|
|
|
|
const dl = document.createElement('a');
|
|
|
|
dl.href = `data:application/octet-stream;base64,${btoa(uint8ToString(tape.out))}`;
|
|
|
|
dl.download = 'rageshake.tar';
|
|
|
|
document.body.appendChild(dl);
|
|
|
|
dl.click();
|
|
|
|
document.body.removeChild(dl);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Source: https://github.com/beatgammit/tar-js/blob/master/examples/main.js
|
|
|
|
function uint8ToString(buf) {
|
|
|
|
let i, length, out = '';
|
|
|
|
for (i = 0, length = buf.length; i < length; i += 1) {
|
|
|
|
out += String.fromCharCode(buf[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2018-04-13 03:34:16 +03:00
|
|
|
function _submitReport(endpoint, body, progressCallback) {
|
2019-11-12 14:40:38 +03:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const req = new XMLHttpRequest();
|
|
|
|
req.open("POST", endpoint);
|
|
|
|
req.timeout = 5 * 60 * 1000;
|
|
|
|
req.onreadystatechange = function() {
|
|
|
|
if (req.readyState === XMLHttpRequest.LOADING) {
|
|
|
|
progressCallback(_t("Waiting for response from server"));
|
|
|
|
} else if (req.readyState === XMLHttpRequest.DONE) {
|
|
|
|
// on done
|
|
|
|
if (req.status < 200 || req.status >= 400) {
|
|
|
|
reject(new Error(`HTTP ${req.status}`));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
req.send(body);
|
|
|
|
});
|
2018-04-13 03:34:16 +03:00
|
|
|
}
|