/** * @fileoverview HTML reporter * @author Julian Laval */ "use strict"; //------------------------------------------------------------------------------ // Helpers //------------------------------------------------------------------------------ const encodeHTML = (function() { const encodeHTMLRules = { "&": "&", "<": "<", ">": ">", '"': """, "'": "'" }; const matchHTML = /[&<>"']/ug; return function(code) { return code ? code.toString().replace(matchHTML, m => encodeHTMLRules[m] || m) : ""; }; }()); /** * Get the final HTML document. * @param {Object} it data for the document. * @returns {string} HTML document. */ function pageTemplate(it) { const { reportColor, reportSummary, date, results } = it; return ` <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>ESLint Report</title> <link rel="icon" type="image/png" sizes="any" href=""> <link rel="icon" type="image/svg+xml" href=""> <style> body { font-family: Arial, "Helvetica Neue", Helvetica, sans-serif; font-size: 16px; font-weight: normal; margin: 0; padding: 0; color: #333; } #overview { padding: 20px 30px; } td, th { padding: 5px 10px; } h1 { margin: 0; } table { margin: 30px; width: calc(100% - 60px); max-width: 1000px; border-radius: 5px; border: 1px solid #ddd; border-spacing: 0; } th { font-weight: 400; font-size: medium; text-align: left; cursor: pointer; } td.clr-1, td.clr-2, th span { font-weight: 700; } th span { float: right; margin-left: 20px; } th span::after { content: ""; clear: both; display: block; } tr:last-child td { border-bottom: none; } tr td:first-child, tr td:last-child { color: #9da0a4; } #overview.bg-0, tr.bg-0 th { color: #468847; background: #dff0d8; border-bottom: 1px solid #d6e9c6; } #overview.bg-1, tr.bg-1 th { color: #f0ad4e; background: #fcf8e3; border-bottom: 1px solid #fbeed5; } #overview.bg-2, tr.bg-2 th { color: #b94a48; background: #f2dede; border-bottom: 1px solid #eed3d7; } td { border-bottom: 1px solid #ddd; } td.clr-1 { color: #f0ad4e; } td.clr-2 { color: #b94a48; } td a { color: #3a33d1; text-decoration: none; } td a:hover { color: #272296; text-decoration: underline; } </style> </head> <body> <div id="overview" class="bg-${reportColor}"> <h1>ESLint Report</h1> <div> <span>${reportSummary}</span> - Generated on ${date} </div> </div> <table> <tbody> ${results} </tbody> </table> <script type="text/javascript"> var groups = document.querySelectorAll("tr[data-group]"); for (i = 0; i < groups.length; i++) { groups[i].addEventListener("click", function() { var inGroup = document.getElementsByClassName(this.getAttribute("data-group")); this.innerHTML = (this.innerHTML.indexOf("+") > -1) ? this.innerHTML.replace("+", "-") : this.innerHTML.replace("-", "+"); for (var j = 0; j < inGroup.length; j++) { inGroup[j].style.display = (inGroup[j].style.display !== "none") ? "none" : "table-row"; } }); } </script> </body> </html> `.trimStart(); } /** * Given a word and a count, append an s if count is not one. * @param {string} word A word in its singular form. * @param {int} count A number controlling whether word should be pluralized. * @returns {string} The original word with an s on the end if count is not one. */ function pluralize(word, count) { return (count === 1 ? word : `${word}s`); } /** * Renders text along the template of x problems (x errors, x warnings) * @param {string} totalErrors Total errors * @param {string} totalWarnings Total warnings * @returns {string} The formatted string, pluralized where necessary */ function renderSummary(totalErrors, totalWarnings) { const totalProblems = totalErrors + totalWarnings; let renderedText = `${totalProblems} ${pluralize("problem", totalProblems)}`; if (totalProblems !== 0) { renderedText += ` (${totalErrors} ${pluralize("error", totalErrors)}, ${totalWarnings} ${pluralize("warning", totalWarnings)})`; } return renderedText; } /** * Get the color based on whether there are errors/warnings... * @param {string} totalErrors Total errors * @param {string} totalWarnings Total warnings * @returns {int} The color code (0 = green, 1 = yellow, 2 = red) */ function renderColor(totalErrors, totalWarnings) { if (totalErrors !== 0) { return 2; } if (totalWarnings !== 0) { return 1; } return 0; } /** * Get HTML (table row) describing a single message. * @param {Object} it data for the message. * @returns {string} HTML (table row) describing the message. */ function messageTemplate(it) { const { parentIndex, lineNumber, columnNumber, severityNumber, severityName, message, ruleUrl, ruleId } = it; return ` <tr style="display: none;" class="f-${parentIndex}"> <td>${lineNumber}:${columnNumber}</td> <td class="clr-${severityNumber}">${severityName}</td> <td>${encodeHTML(message)}</td> <td> <a href="${ruleUrl ? ruleUrl : ""}" target="_blank" rel="noopener noreferrer">${ruleId ? ruleId : ""}</a> </td> </tr> `.trimStart(); } /** * Get HTML (table rows) describing the messages. * @param {Array} messages Messages. * @param {int} parentIndex Index of the parent HTML row. * @param {Object} rulesMeta Dictionary containing metadata for each rule executed by the analysis. * @returns {string} HTML (table rows) describing the messages. */ function renderMessages(messages, parentIndex, rulesMeta) { /** * Get HTML (table row) describing a message. * @param {Object} message Message. * @returns {string} HTML (table row) describing a message. */ return messages.map(message => { const lineNumber = message.line || 0; const columnNumber = message.column || 0; let ruleUrl; if (rulesMeta) { const meta = rulesMeta[message.ruleId]; if (meta && meta.docs && meta.docs.url) { ruleUrl = meta.docs.url; } } return messageTemplate({ parentIndex, lineNumber, columnNumber, severityNumber: message.severity, severityName: message.severity === 1 ? "Warning" : "Error", message: message.message, ruleId: message.ruleId, ruleUrl }); }).join("\n"); } /** * Get HTML (table row) describing the result for a single file. * @param {Object} it data for the file. * @returns {string} HTML (table row) describing the result for the file. */ function resultTemplate(it) { const { color, index, filePath, summary } = it; return ` <tr class="bg-${color}" data-group="f-${index}"> <th colspan="4"> [+] ${encodeHTML(filePath)} <span>${encodeHTML(summary)}</span> </th> </tr> `.trimStart(); } /** * Render the results. * @param {Array} results Test results. * @param {Object} rulesMeta Dictionary containing metadata for each rule executed by the analysis. * @returns {string} HTML string describing the results. */ function renderResults(results, rulesMeta) { return results.map((result, index) => resultTemplate({ index, color: renderColor(result.errorCount, result.warningCount), filePath: result.filePath, summary: renderSummary(result.errorCount, result.warningCount) }) + renderMessages(result.messages, index, rulesMeta)).join("\n"); } //------------------------------------------------------------------------------ // Public Interface //------------------------------------------------------------------------------ module.exports = function(results, data) { let totalErrors, totalWarnings; const metaData = data ? data.rulesMeta : {}; totalErrors = 0; totalWarnings = 0; // Iterate over results to get totals results.forEach(result => { totalErrors += result.errorCount; totalWarnings += result.warningCount; }); return pageTemplate({ date: new Date(), reportColor: renderColor(totalErrors, totalWarnings), reportSummary: renderSummary(totalErrors, totalWarnings), results: renderResults(results, metaData) }); };