/* Copyright 2017 OpenMarket Ltd Copyright 2018 New Vector Ltd Copyright 2019 The Matrix.org Foundation C.I.C. 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'; import {MatrixClientPeg} from '../MatrixClientPeg'; import PlatformPeg from '../PlatformPeg'; import { _t } from '../languageHandler'; import * as rageshake from './rageshake'; // polyfill textencoder if necessary import * as TextEncodingUtf8 from 'text-encoding-utf-8'; import SettingsStore from "../settings/SettingsStore"; let TextEncoder = window.TextEncoder; if (!TextEncoder) { TextEncoder = TextEncodingUtf8.TextEncoder; } /** * 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 || (() => {}); progressCallback(_t("Collecting app version information")); let version = "UNKNOWN"; try { version = await PlatformPeg.get().getAppVersion(); } catch (err) {} // PlatformPeg already logs this. let userAgent = "UNKNOWN"; if (window.navigator && window.navigator.userAgent) { userAgent = window.navigator.userAgent; } let installedPWA = "UNKNOWN"; try { // Known to work at least for desktop Chrome installedPWA = window.matchMedia('(display-mode: standalone)').matches; } catch (e) { } let touchInput = "UNKNOWN"; try { // MDN claims broad support across browsers touchInput = window.matchMedia('(pointer: coarse)').matches; } catch (e) { } 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); body.append('installed_pwa', installedPWA); body.append('touch_input', touchInput); if (client) { body.append('user_id', client.credentials.userId); body.append('device_id', client.deviceId); } if (client.isCryptoEnabled()) { 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()); } if (opts.label) { body.append('label', opts.label); } // add labs options const enabledLabs = SettingsStore.getLabsFeatures().filter(SettingsStore.isFeatureEnabled); if (enabledLabs.length) { body.append('enabled_labs', enabledLabs.join(', ')); } // add storage persistence/quota information if (navigator.storage && navigator.storage.persisted) { try { body.append("storageManager_persisted", await navigator.storage.persisted()); } catch (e) {} } else if (document.hasStorageAccess) { // Safari try { body.append("storageManager_persisted", await document.hasStorageAccess()); } catch (e) {} } 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) {} } if (window.Modernizr) { const missingFeatures = Object.keys(window.Modernizr).filter(key => window.Modernizr[key] === false); if (missingFeatures.length > 0) { body.append("modernizr_missing_features", missingFeatures.join(", ")); } } if (opts.sendLogs) { progressCallback(_t("Collecting logs")); const logs = await rageshake.getLogsForReport(); for (const entry of logs) { // 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); } } progressCallback(_t("Uploading report")); await _submitReport(bugReportEndpoint, body, progressCallback); } function _submitReport(endpoint, body, progressCallback) { 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); }); }