| /** | |
|  * @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) | |
|     }); | |
| };
 |