From fc40b921ca57f00fb0e464b60c33b7525724b40a Mon Sep 17 00:00:00 2001 From: Adam Bradley Date: Thu, 10 Nov 2016 17:07:45 -0600 Subject: [PATCH] feat(error): client runtime error reporting --- bin/ion-dev.js | 245 ++++++++++++++------------ src/dev-server/notification-server.ts | 79 +++++---- src/util/logger-diagnostics.ts | 172 +++++++++++++++--- 3 files changed, 330 insertions(+), 166 deletions(-) diff --git a/bin/ion-dev.js b/bin/ion-dev.js index 90829eac..0579d32e 100644 --- a/bin/ion-dev.js +++ b/bin/ion-dev.js @@ -1,4 +1,5 @@ window.IonicDevServerConfig = window.IonicDevServerConfig || {}; + window.IonicDevServer = { start: function() { this.msgQueue = []; @@ -15,132 +16,94 @@ window.IonicDevServer = { } this.openConnection(); - this.bindKeyboardEvents(); - document.addEventListener("DOMContentLoaded", IonicDevServer.domReady); - }, - - domReady: function() { - document.removeEventListener("DOMContentLoaded", IonicDevServer.domReady); - var diagnosticsEle = document.getElementById('ion-diagnostics'); - if (diagnosticsEle) { - IonicDevServer.buildStatus('error'); - } else { - IonicDevServer.buildStatus('success'); - } - }, - - handleError: function(err) { - var self = this; - - console.error('Handling error', err); - - var existing = document.querySelector('._ionic-error-view'); - if(existing) { - document.body.removeChild(existing); - } - - this._errorWindow = this._makeErrorWindow(err); - document.body.appendChild(this._errorWindow); - - setTimeout(function() { - window.requestAnimationFrame(function() { - self._errorWindow.classList.add('show'); - }); - }, 500); - - }, - - _closeErrorWindow: function() { var self = this; - window.requestAnimationFrame(function() { - self._errorWindow.classList.remove('show'); - setTimeout(function() { - document.body.removeChild(self._errorWindow); - self._errorWindow = null; - }, 500); + document.addEventListener("DOMContentLoaded", function() { + var diagnosticsEle = document.getElementById('ion-diagnostics'); + if (diagnosticsEle) { + self.buildStatus('error'); + } else { + self.buildStatus('success'); + } }); }, - _makeErrorWindow: function(err) { - var self = this; - - var isInCordova = !!window.cordova; + handleError: function(err) { + if (!err) return; - var d = document.createElement('div'); - d.className = '_ionic-error-view'; + if (this.socketReady) { + var msg = { + category: 'runtimeError', + type: 'runtimeError', + data: { + message: err.message ? err.message.toString() : null, + stack: err.stack ? err.stack.toString() : null + } + }; + this.queueMessageSend(msg); - if(isInCordova) { - d.classList.add('_ionic-error-in-cordova'); + } else { + var c = []; + + c.push('
'); + c.push('
Error
'); + c.push('
'); + c.push(''); + c.push('
'); + c.push('
'); + + c.push('
'); + c.push('
'); + c.push('
Runtime Error
'); + c.push('
' + err.message + '
'); + c.push('
'); + c.push('
Stack
'); + c.push('
' + err.stack + '
'); + c.push('
'); + + this.buildUpdate({ + type: 'clientError', + data: { + diagnosticsHtml: c.join('') + } + }); } - - d.innerHTML = '

App Runtime Error

Close
' + err.message + '

Stacktrace

' + this._makeErrorButtonsHtml() + '
'; - - d.querySelector('._close').addEventListener('click', function(e) { - closeWindow(d); - }); - /* - d.querySelector('[action="copy"]').addEventListener('click', function(e) { - if(window.IonicDevtools) { - window.IonicDevtools.copyErrorToClipboard(err); - } - }); - */ - - return d; }, - _makeErrorButtonsHtml: function() { - var d = document.createElement('div'); - d.className = '_ion-error-buttons'; - - var b1 = document.createElement('button'); - b1.innerHTML = 'Close (ESC)'; - b1.className = '_button'; - - var b2 = document.createElement('button'); - b2.innerHTML = 'Reload (⌘)'; - b2.className = '_button'; - //d.appendChild(b1); - //d.appendChild(b2); - - return d.innerHTML; - }, reloadApp: function() { - if(window.cordova) { - window.location.reload(true); - } - }, - showDebugMenu: function() { - if(window.IonicDevtools) { - window.IonicDevtools.showDebugMenu(); - } + window.location.reload(true); }, + bindKeyboardEvents: function() { var self = this; - document.addEventListener('keyup', function(event) { - var key = event.keyCode || event.charCode || 0; + document.addEventListener('keyup', function(ev) { + var key = ev.keyCode || ev.charCode || 0; - if(key == 27 && self._errorWindow) { + if (key == 27 && self._errorWindow) { self._closeErrorWindow(); } }); - document.addEventListener('keydown', function(event) { - var key = event.keyCode || event.charCode || 0; + + document.addEventListener('keydown', function(ev) { + var key = ev.keyCode || ev.charCode || 0; // Check for reload command (cmd/ctrl+R) - if(key == 82 && (event.metaKey || event.ctrlKey)) { + if (key == 82 && (ev.metaKey || ev.ctrlKey)) { self.reloadApp(); } + }); - // Check for debugger command (cmd/ctrl+D) - /* - if(key == 68 && (event.metaKey || event.ctrlKey)) { - self.showDebugMenu(); + document.addEventListener('click', function(ev) { + if (ev.target && ev.target.classList.contains('ion-diagnostic-close')) { + self.buildUpdate({ + type: 'closeDiagnostics', + data: { + diagnosticsHtml: null + } + }); } - */ }); }, @@ -164,8 +127,9 @@ window.IonicDevServer = { } }; - self.socket.onclose = () => { + self.socket.onclose = function() { self.consoleLog('Dev server logger closed'); + self.socketReady = false; }; self.drainMessageQueue(); @@ -184,7 +148,7 @@ window.IonicDevServer = { try { this.socket.send(JSON.stringify(msg)); } catch(e) { - if(e instanceof TypeError) { + if (e instanceof TypeError) { } else { this.consoleError('ws error: ' + e); @@ -243,7 +207,8 @@ window.IonicDevServer = { toastEle.innerHTML = c.join(''); document.body.insertBefore(toastEle, document.body.firstChild); } - IonicDevServer.toastTimerId = setTimeout(function() { + + this.toastTimerId = setTimeout(function() { var toastEle = document.getElementById('ion-diagnostics-toast'); if (toastEle) { toastEle.classList.add('ion-diagnostics-toast-active'); @@ -253,7 +218,7 @@ window.IonicDevServer = { } else { status = msg.data.diagnosticsHtml ? 'error' : 'success'; - clearTimeout(IonicDevServer.toastTimerId); + clearTimeout(this.toastTimerId); var toastEle = document.getElementById('ion-diagnostics-toast'); if (toastEle) { @@ -263,7 +228,8 @@ window.IonicDevServer = { var diagnosticsEle = document.getElementById('ion-diagnostics'); if (diagnosticsEle && !msg.data.diagnosticsHtml) { diagnosticsEle.classList.add('ion-diagnostics-fade-out'); - IonicDevServer.diagnosticsTimerId = setTimeout(function() { + + this.diagnosticsTimerId = setTimeout(function() { var diagnosticsEle = document.getElementById('ion-diagnostics'); if (diagnosticsEle) { diagnosticsEle.parentElement.removeChild(diagnosticsEle); @@ -271,25 +237,27 @@ window.IonicDevServer = { }, 100); } else if (msg.data.diagnosticsHtml) { - clearTimeout(IonicDevServer.diagnosticsTimerId); + + clearTimeout(this.diagnosticsTimerId); if (!diagnosticsEle) { diagnosticsEle = document.createElement('div'); diagnosticsEle.id = 'ion-diagnostics'; diagnosticsEle.className = 'ion-diagnostics-fade-out'; document.body.insertBefore(diagnosticsEle, document.body.firstChild); - IonicDevServer.diagnosticsTimerId = setTimeout(function() { + + this.diagnosticsTimerId = setTimeout(function() { var diagnosticsEle = document.getElementById('ion-diagnostics'); if (diagnosticsEle) { diagnosticsEle.classList.remove('ion-diagnostics-fade-out'); } }, 24); } - diagnosticsEle.innerHTML = msg.data.diagnosticsHtml; + diagnosticsEle.innerHTML = msg.data.diagnosticsHtml } } - IonicDevServer.buildStatus(status); + this.buildStatus(status); }, buildStatus: function (status) { @@ -301,7 +269,7 @@ window.IonicDevServer = { var iconLink = document.createElement('link'); iconLink.rel = 'icon'; iconLink.type = 'image/png'; - iconLink.href = IonicDevServer[status + 'Icon']; + iconLink.href = this[status + 'Icon']; document.head.appendChild(iconLink); if (status === 'error') { @@ -309,7 +277,7 @@ window.IonicDevServer = { if (diagnosticsEle) { var systemInfoEle = diagnosticsEle.querySelector('#ion-diagnostics-system-info'); if (!systemInfoEle) { - systemInfoEle = document.createElement('pre'); + systemInfoEle = document.createElement('div'); systemInfoEle.id = 'ion-diagnostics-system-info'; systemInfoEle.innerHTML = IonicDevServerConfig.systemInfo.join('\n'); diagnosticsEle.appendChild(systemInfoEle); @@ -318,6 +286,61 @@ window.IonicDevServer = { } }, + showOptions: function() { + + }, + + enableShake: function() { + /* + * Author: Alex Gibson + * https://github.com/alexgibson/shake.js + * License: MIT license + */ + var self = this; + var threshold = 15; + var timeout = 1000; + + self.shakeTime = new Date(); + self.shakeX = null; + self.shakeY = null; + self.shakeZ = null; + + window.addEventListener('devicemotion', function(ev) { + var current = ev.accelerationIncludingGravity; + var currentTime; + var timeDifference; + var deltaX = 0; + var deltaY = 0; + var deltaZ = 0; + + if (self.shakeX === null) { + self.shakeX = current.x; + self.shakeY = current.y; + self.shakeZ = current.z; + return; + } + + deltaX = Math.abs(self.shakeX - current.x); + deltaY = Math.abs(self.shakeY - current.y); + deltaZ = Math.abs(self.shakeZ - current.z); + + if (((deltaX > threshold) && (deltaY > threshold)) || ((deltaX > threshold) && (deltaZ > threshold)) || ((deltaY > threshold) && (deltaZ > threshold))) { + currentTime = new Date(); + timeDifference = currentTime.getTime() - self.shakeTime.getTime(); + + if (timeDifference > timeout) { + self.showOptions(); + self.shakeTime = new Date(); + } + } + + self.shakeX = current.x; + self.shakeY = current.y; + self.shakeZ = current.z; + }); + + }, + activeIcon: '', errorIcon: '', diff --git a/src/dev-server/notification-server.ts b/src/dev-server/notification-server.ts index a8a1542f..465668f6 100644 --- a/src/dev-server/notification-server.ts +++ b/src/dev-server/notification-server.ts @@ -1,6 +1,6 @@ // Ionic Dev Server: Server Side Logger import { Logger } from '../util/logger'; -import { hasDiagnostics, getDiagnosticsHtmlContent } from '../util/logger-diagnostics'; +import { hasDiagnostics, getDiagnosticsHtmlContent, generateRuntimeDiagnosticContent } from '../util/logger-diagnostics'; import { on, EventType } from '../util/events'; import { Server as WebSocketServer } from 'ws'; import { ServeConfig } from './serve-config'; @@ -8,6 +8,7 @@ import { ServeConfig } from './serve-config'; export function createNotificationServer(config: ServeConfig) { let wsServer: any; + const msgToClient: WsMessage[] = []; // queue up all messages to the client function queueMessageSend(msg: WsMessage) { @@ -75,51 +76,61 @@ export function createNotificationServer(config: ServeConfig) { drainMessageQueue(); }); -} - -function printMessageFromClient(msg: WsMessage) { - if (msg.data) { - switch (msg.category) { - case 'console': - printConsole(msg); - break; + function printMessageFromClient(msg: WsMessage) { + if (msg && msg.data) { + switch (msg.category) { + case 'console': + printConsole(msg); + break; - case 'exception': - printException(msg); - break; + case 'runtimeError': + handleRuntimeError(msg); + break; + } } } -} -function printConsole(msg: WsMessage) { - const args = msg.data; - args[0] = `console.${msg.type}: ${args[0]}`; - switch (msg.type) { - case 'error': - Logger.error.apply(this, args); - break; + function printConsole(msg: WsMessage) { + const args = msg.data; + args[0] = `console.${msg.type}: ${args[0]}`; - case 'warn': - Logger.warn.apply(this, args); - break; + switch (msg.type) { + case 'error': + Logger.error.apply(this, args); + break; - case 'debug': - Logger.debug.apply(this, args); - break; + case 'warn': + Logger.warn.apply(this, args); + break; - default: - Logger.info.apply(this, args); - break; - } -} + case 'debug': + Logger.debug.apply(this, args); + break; -function printException(msg: WsMessage) { + default: + Logger.info.apply(this, args); + break; + } + } -} -const msgToClient: WsMessage[] = []; + function handleRuntimeError(clientMsg: WsMessage) { + const msg: WsMessage = { + category: 'buildUpdate', + type: 'completed', + data: { + diagnosticsHtml: generateRuntimeDiagnosticContent(config.rootDir, + config.buildDir, + clientMsg.data.message, + clientMsg.data.stack) + } + }; + queueMessageSend(msg); + } + +} export interface WsMessage { category: string; diff --git a/src/util/logger-diagnostics.ts b/src/util/logger-diagnostics.ts index f7e0a377..4b2eb998 100644 --- a/src/util/logger-diagnostics.ts +++ b/src/util/logger-diagnostics.ts @@ -1,8 +1,8 @@ import { BuildContext } from './interfaces'; import { Diagnostic, Logger, PrintLine } from './logger'; -import { titleCase } from './helpers'; -import { join } from 'path'; +import { join, resolve , normalize} from 'path'; import { readFileSync, unlinkSync, writeFileSync } from 'fs'; +import { splitLineBreaks, titleCase } from './helpers'; import * as chalk from 'chalk'; @@ -116,7 +116,7 @@ export function clearDiagnostics(context: BuildContext, type: string) { export function hasDiagnostics(buildDir: string) { - loadDiagnosticsHtml(buildDir); + loadBuildDiagnosticsHtml(buildDir); const keys = Object.keys(diagnosticsHtmlCache); for (var i = 0; i < keys.length; i++) { @@ -129,7 +129,7 @@ export function hasDiagnostics(buildDir: string) { } -function loadDiagnosticsHtml(buildDir: string) { +function loadBuildDiagnosticsHtml(buildDir: string) { try { if (diagnosticsHtmlCache[DiagnosticsType.TypeScript] === undefined) { diagnosticsHtmlCache[DiagnosticsType.TypeScript] = readFileSync(getDiagnosticsFileName(buildDir, DiagnosticsType.TypeScript), 'utf8'); @@ -155,35 +155,52 @@ export function injectDiagnosticsHtml(buildDir: string, content: any) { let contentStr = content.toString(); - const diagnosticsHtml: string[] = []; - diagnosticsHtml.push(`
`); - diagnosticsHtml.push(getDiagnosticsHtmlContent(buildDir)); - diagnosticsHtml.push(`
`); + const c: string[] = []; + c.push(`
`); + + // diagnostics content + c.push(getDiagnosticsHtmlContent(buildDir)); + + c.push(`
`); // #ion-diagnostics let match = contentStr.match(/(?![\s\S]*)/i); if (match) { - contentStr = contentStr.replace(match[0], match[0] + '\n' + diagnosticsHtml.join('\n')); + contentStr = contentStr.replace(match[0], match[0] + '\n' + c.join('\n')); } else { - contentStr = diagnosticsHtml.join('\n') + contentStr; + contentStr = c.join('\n') + contentStr; } return contentStr; } -export function getDiagnosticsHtmlContent(buildDir: string) { - loadDiagnosticsHtml(buildDir); +export function getDiagnosticsHtmlContent(buildDir: string, includeDiagnosticsHtml?: string) { + const c: string[] = []; + + // diagnostics header + c.push(` +
+
Error
+
+ +
+
+ `); + + if (includeDiagnosticsHtml) { + c.push(includeDiagnosticsHtml); + } - const diagnosticsHtml: string[] = []; + loadBuildDiagnosticsHtml(buildDir); const keys = Object.keys(diagnosticsHtmlCache); for (var i = 0; i < keys.length; i++) { if (typeof diagnosticsHtmlCache[keys[i]] === 'string') { - diagnosticsHtml.push(diagnosticsHtmlCache[keys[i]]); + c.push(diagnosticsHtmlCache[keys[i]]); } } - return diagnosticsHtml.join('\n'); + return c.join('\n'); } @@ -194,13 +211,24 @@ function generateDiagnosticHtml(d: Diagnostic) { c.push(`
`); - const header = `${titleCase(d.type)} ${titleCase(d.level)}`; - c.push(`
${escapeHtml(header)}
`); + const title = `${titleCase(d.type)} ${titleCase(d.level)}`; + c.push(`
${escapeHtml(title)}
`); c.push(`
${escapeHtml(d.messageText)}
`); c.push(`
`); // .ion-diagnostic-masthead + c.push(generateCodeBlock(d)); + + c.push(``); // .ion-diagnostic + + return c.join('\n'); +} + + +function generateCodeBlock(d: Diagnostic) { + const c: string[] = []; + c.push(`
`); c.push(`
${escapeHtml(d.relFileName)}
`); @@ -238,12 +266,114 @@ function generateDiagnosticHtml(d: Diagnostic) { c.push(`
`); // .ion-diagnostic-file - c.push(``); // .ion-diagnostic - return c.join('\n'); } +export function generateRuntimeDiagnosticContent(rootDir: string, buildDir: string, runtimeErrorMessage: string, runtimeErrorStack: string) { + let c: string[] = []; + + c.push('
'); + c.push('
'); + c.push('
Runtime Error
'); + if (runtimeErrorMessage) { + c.push('
' + escapeHtml(runtimeErrorMessage) + '
'); + } + c.push('
'); // .ion-diagnostic-masthead + + const diagnosticsHtmlCache = generateRuntimeStackDiagnostics(rootDir, runtimeErrorStack); + diagnosticsHtmlCache.forEach(d => { + c.push(generateCodeBlock(d)); + }); + + if (runtimeErrorStack) { + c.push('
Stack
'); + c.push('
' + escapeHtml(runtimeErrorStack) + '
'); + } + + c.push('
'); // .ion-diagnostic + + return getDiagnosticsHtmlContent(buildDir, c.join('\n')); +} + + +export function generateRuntimeStackDiagnostics(rootDir: string, stack: string) { + const diagnostics: Diagnostic[] = []; + + if (stack) { + splitLineBreaks(stack).forEach(stackLine => { + try { + const match = WEBPACK_FILE_REGEX.exec(stackLine); + if (!match) return; + + const fileSplit = match[1].split('?'); + if (fileSplit.length !== 2) return; + + const linesSplit = fileSplit[1].split(':'); + if (linesSplit.length !== 3) return; + + const fileName = fileSplit[0]; + const errorLineIndex = parseInt(linesSplit[1], 10); + const errorCharIndex = parseInt(linesSplit[2], 10); + + const d: Diagnostic = { + level: 'error', + syntax: 'js', + type: 'runtime', + header: '', + code: 'runtime', + messageText: '', + absFileName: resolve(rootDir, fileName), + relFileName: normalize(fileName), + lines: [] + }; + + const srcLines = splitLineBreaks(readFileSync(d.absFileName, 'utf8')); + if (!srcLines.length || errorLineIndex >= srcLines.length) return; + + const errorLine: PrintLine = { + lineIndex: errorLineIndex, + lineNumber: errorLineIndex + 1, + text: srcLines[errorLineIndex], + errorCharStart: errorCharIndex, + errorLength: 1 + }; + d.lines.push(errorLine); + + if (errorLine.lineIndex > 0) { + const beforeLine: PrintLine = { + lineIndex: errorLine.lineIndex - 1, + lineNumber: errorLine.lineNumber - 1, + text: srcLines[errorLine.lineIndex - 1], + errorCharStart: -1, + errorLength: -1 + }; + d.lines.unshift(beforeLine); + } + + if (errorLine.lineIndex < srcLines.length) { + const beforeLine: PrintLine = { + lineIndex: errorLine.lineIndex + 1, + lineNumber: errorLine.lineNumber + 1, + text: srcLines[errorLine.lineIndex + 1], + errorCharStart: -1, + errorLength: -1 + }; + d.lines.push(beforeLine); + } + + diagnostics.push(d); + + } catch (e) {} + }); + } + + return diagnostics; +}; + +const WEBPACK_FILE_REGEX = /\(webpack:\/\/\/(.*?)\)/; + + function htmlHighlightError(errorLine: string, errorCharStart: number, errorLength: number) { const lineChars: string[] = []; const lineLength = Math.max(errorLine.length, errorCharStart + errorLength); @@ -301,6 +431,7 @@ function cssConsoleSyntaxHighlight(text: string, errorCharStart: number) { return chars.join(''); } + function escapeHtml(unsafe: string) { return unsafe .replace(/&/g, '&') @@ -308,8 +439,7 @@ function escapeHtml(unsafe: string) { .replace(/>/g, '>') .replace(/"/g, '"') .replace(/'/g, '''); - } - +} function removeWhitespaceIndent(orgLines: PrintLine[]) {