192 lines
5.7 KiB
TypeScript
192 lines
5.7 KiB
TypeScript
// Copyright 2015-2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import copyText from 'copy-text-to-clipboard';
|
|
import * as log from '../logging/log';
|
|
import * as debugLog from '../logging/debuglogs';
|
|
import { ToastLoadingFullLogs } from '../components/ToastLoadingFullLogs';
|
|
import { ToastLinkCopied } from '../components/ToastLinkCopied';
|
|
import { showToast } from '../util/showToast';
|
|
|
|
window.Whisper = window.Whisper || {};
|
|
const { Whisper } = window;
|
|
|
|
// This enum-like object describes the load state of `DebugLogView`. It's designed to be
|
|
// unidirectional; `NotStarted` → `Started` → `LogsFetchedButNotInTextarea`, etc.
|
|
const LoadState = {
|
|
NotStarted: 0,
|
|
Started: 1,
|
|
LogsFetchedButNotInTextarea: 2,
|
|
PuttingLogsInTextarea: 3,
|
|
LogsInTextarea: 4,
|
|
};
|
|
|
|
const DebugLogLinkView = Whisper.View.extend({
|
|
template: () => $('#debug-log-link').html(),
|
|
initialize(options: { url: string }) {
|
|
this.url = options.url;
|
|
},
|
|
events: {
|
|
'click .copy': 'copy',
|
|
},
|
|
render_attributes() {
|
|
return {
|
|
url: this.url,
|
|
reportIssue: window.i18n('reportIssue'),
|
|
debugLogCopy: window.i18n('debugLogCopy'),
|
|
debugLogCopyAlt: window.i18n('debugLogCopyAlt'),
|
|
};
|
|
},
|
|
copy(e: MouseEvent) {
|
|
e.preventDefault();
|
|
const target = e.currentTarget as HTMLAnchorElement;
|
|
copyText(target.href);
|
|
showToast(ToastLinkCopied);
|
|
},
|
|
});
|
|
|
|
/**
|
|
* The bulk of the logic in this view involves grabbing the logs from disk and putting
|
|
* them in a `<textarea>`. The first part isn't instant but is reasonably fast; setting
|
|
* the textarea's `value` takes a long time.
|
|
*
|
|
* After loading the logs into memory, we only put a small number of lines into the
|
|
* textarea. If the user clicks or scrolls the textarea, we pull the full logs, which
|
|
* can cause the system to lock up for a bit.
|
|
*
|
|
* Ideally, we'd only show a sampling of the logs and allow the user to download and
|
|
* edit them in their own editor. This is mostly a stopgap solution.
|
|
*/
|
|
export const DebugLogView = Whisper.View.extend({
|
|
template: () => $('#debug-log').html(),
|
|
className: 'debug-log modal',
|
|
initialize() {
|
|
this.render();
|
|
|
|
this.textarea = this.$('.textarea').get(0);
|
|
if (!this.textarea) {
|
|
throw new Error('textarea not found');
|
|
}
|
|
this.textarea.setAttribute('readonly', '');
|
|
|
|
this.loadState = LoadState.NotStarted;
|
|
this.putFullLogsInTextareaPlease = false;
|
|
|
|
this.fetchLogs();
|
|
},
|
|
events: {
|
|
'click .textarea': 'putFullLogsInTextarea',
|
|
'scroll .textarea': 'putFullLogsInTextarea',
|
|
'wheel .textarea': 'putFullLogsInTextarea',
|
|
'click .submit': 'submit',
|
|
'click .close': 'close',
|
|
},
|
|
render_attributes: {
|
|
title: window.i18n('submitDebugLog'),
|
|
cancel: window.i18n('cancel'),
|
|
submit: window.i18n('submit'),
|
|
close: window.i18n('gotIt'),
|
|
debugLogExplanation: window.i18n('debugLogExplanation'),
|
|
},
|
|
async fetchLogs() {
|
|
if (this.loadState !== LoadState.NotStarted) {
|
|
return;
|
|
}
|
|
|
|
this.loadState = LoadState.Started;
|
|
this.textarea.value = window.i18n('loading');
|
|
this.$('.submit').attr('disabled', 'disabled');
|
|
|
|
this.logText = await debugLog.fetch();
|
|
this.loadState = LoadState.LogsFetchedButNotInTextarea;
|
|
|
|
// This number is somewhat arbitrary; we want to show enough that it's clear that
|
|
// we need to scroll, but not so many that things get slow.
|
|
const linesToShow = Math.ceil(Math.min(window.innerHeight, 2000) / 5);
|
|
this.textarea.value = this.logText
|
|
.split(/\n/g, linesToShow)
|
|
.concat(['', window.i18n('loading')])
|
|
.join('\n');
|
|
|
|
this.$('.submit').removeAttr('disabled');
|
|
|
|
if (this.putFullLogsInTextareaPlease) {
|
|
this.putFullLogsInTextarea();
|
|
}
|
|
},
|
|
putFullLogsInTextarea() {
|
|
switch (this.loadState) {
|
|
case LoadState.NotStarted:
|
|
case LoadState.Started:
|
|
this.putFullLogsInTextareaPlease = true;
|
|
break;
|
|
case LoadState.LogsInTextarea:
|
|
case LoadState.PuttingLogsInTextarea:
|
|
break;
|
|
case LoadState.LogsFetchedButNotInTextarea:
|
|
if (!this.logText) {
|
|
throw new Error('Expected log text to be present');
|
|
}
|
|
this.loadState = LoadState.PuttingLogsInTextarea;
|
|
showToast(ToastLoadingFullLogs);
|
|
setTimeout(() => {
|
|
this.textarea.value = this.logText;
|
|
this.textarea.removeAttribute('readonly');
|
|
this.loadState = LoadState.LogsInTextarea;
|
|
}, 0);
|
|
break;
|
|
default:
|
|
// When we can, we should make this throw a `missingCaseError`.
|
|
break;
|
|
}
|
|
},
|
|
close() {
|
|
window.closeDebugLog();
|
|
},
|
|
async submit(e: SubmitEvent) {
|
|
e.preventDefault();
|
|
|
|
let text;
|
|
switch (this.loadState) {
|
|
case LoadState.NotStarted:
|
|
case LoadState.Started:
|
|
return;
|
|
case LoadState.LogsFetchedButNotInTextarea:
|
|
text = this.logText;
|
|
break;
|
|
case LoadState.LogsInTextarea:
|
|
text = this.textarea.value;
|
|
break;
|
|
default:
|
|
// When we can, we should make this throw a `missingCaseError`.
|
|
return;
|
|
}
|
|
|
|
if (text.length === 0) {
|
|
return;
|
|
}
|
|
|
|
this.$('.buttons, .textarea').remove();
|
|
this.$('.result').addClass('loading');
|
|
|
|
try {
|
|
const publishedLogURL = await debugLog.upload(text, window.getVersion());
|
|
const view = new DebugLogLinkView({
|
|
url: publishedLogURL,
|
|
el: this.$('.result'),
|
|
});
|
|
this.$('.loading').removeClass('loading');
|
|
view.render();
|
|
this.$('.link').focus().select();
|
|
} catch (error) {
|
|
log.error(
|
|
'DebugLogView error:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
this.$('.loading').removeClass('loading');
|
|
this.$('.result').text(window.i18n('debugLogError'));
|
|
}
|
|
},
|
|
});
|
|
|
|
window.Whisper.DebugLogView = DebugLogView;
|