index.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. 'use strict';
  2. /*
  3. Copyright 2012-2015, Yahoo Inc.
  4. Copyrights licensed under the New BSD License. See the accompanying LICENSE file for terms.
  5. */
  6. const fs = require('fs');
  7. const path = require('path');
  8. const { ReportBase } = require('istanbul-lib-report');
  9. const HtmlReport = require('../html');
  10. const standardLinkMapper = {
  11. getPath(node) {
  12. if (typeof node === 'string') {
  13. return node;
  14. }
  15. let filePath = node.getQualifiedName();
  16. if (node.isSummary()) {
  17. if (filePath !== '') {
  18. filePath += '/index.html';
  19. } else {
  20. filePath = 'index.html';
  21. }
  22. } else {
  23. filePath += '.html';
  24. }
  25. return filePath;
  26. },
  27. relativePath(source, target) {
  28. const targetPath = this.getPath(target);
  29. const sourcePath = path.dirname(this.getPath(source));
  30. return path.relative(sourcePath, targetPath);
  31. },
  32. assetPath(node, name) {
  33. return this.relativePath(this.getPath(node), name);
  34. }
  35. };
  36. class HtmlSpaReport extends ReportBase {
  37. constructor(opts = {}) {
  38. super({
  39. // force the summarizer to nested for html-spa
  40. summarizer: 'nested'
  41. });
  42. this.verbose = opts.verbose || false;
  43. this.linkMapper = opts.linkMapper || standardLinkMapper;
  44. this.subdir = opts.subdir || '';
  45. this.date = Date();
  46. this.skipEmpty = opts.skipEmpty;
  47. this.htmlReport = new HtmlReport(opts);
  48. this.htmlReport.getBreadcrumbHtml = function() {
  49. return '<a href="javascript:history.back()">Back</a>';
  50. };
  51. this.metricsToShow = opts.metricsToShow || [
  52. 'lines',
  53. 'branches',
  54. 'functions'
  55. ];
  56. }
  57. getWriter(context) {
  58. if (!this.subdir) {
  59. return context.writer;
  60. }
  61. return context.writer.writerForDir(this.subdir);
  62. }
  63. onStart(root, context) {
  64. this.htmlReport.onStart(root, context);
  65. const writer = this.getWriter(context);
  66. const srcDir = path.resolve(__dirname, './assets');
  67. fs.readdirSync(srcDir).forEach(f => {
  68. const resolvedSource = path.resolve(srcDir, f);
  69. const resolvedDestination = '.';
  70. const stat = fs.statSync(resolvedSource);
  71. let dest;
  72. if (stat.isFile()) {
  73. dest = resolvedDestination + '/' + f;
  74. if (this.verbose) {
  75. console.log('Write asset: ' + dest);
  76. }
  77. writer.copyFile(resolvedSource, dest);
  78. }
  79. });
  80. }
  81. onDetail(node, context) {
  82. this.htmlReport.onDetail(node, context);
  83. }
  84. getMetric(metric, type, context) {
  85. const isEmpty = metric.total === 0;
  86. return {
  87. total: metric.total,
  88. covered: metric.covered,
  89. skipped: metric.skipped,
  90. missed: metric.total - metric.covered,
  91. pct: isEmpty ? 0 : metric.pct,
  92. classForPercent: isEmpty
  93. ? 'empty'
  94. : context.classForPercent(type, metric.pct)
  95. };
  96. }
  97. toDataStructure(node, context) {
  98. const coverageSummary = node.getCoverageSummary();
  99. const metrics = {
  100. statements: this.getMetric(
  101. coverageSummary.statements,
  102. 'statements',
  103. context
  104. ),
  105. branches: this.getMetric(
  106. coverageSummary.branches,
  107. 'branches',
  108. context
  109. ),
  110. functions: this.getMetric(
  111. coverageSummary.functions,
  112. 'functions',
  113. context
  114. ),
  115. lines: this.getMetric(coverageSummary.lines, 'lines', context)
  116. };
  117. return {
  118. file: node.getRelativeName(),
  119. isEmpty: coverageSummary.isEmpty(),
  120. metrics,
  121. children:
  122. node.isSummary() &&
  123. node
  124. .getChildren()
  125. .map(child => this.toDataStructure(child, context))
  126. };
  127. }
  128. onEnd(rootNode, context) {
  129. const data = this.toDataStructure(rootNode, context);
  130. const cw = this.getWriter(context).writeFile(
  131. this.linkMapper.getPath(rootNode)
  132. );
  133. cw.write(
  134. `<!doctype html>
  135. <html lang="en">
  136. <head>
  137. <link rel="stylesheet" href="spa.css" />
  138. <meta name="viewport" content="width=device-width, initial-scale=1">
  139. </head>
  140. <body>
  141. <div id="app" class="app"></div>
  142. <script>
  143. window.data = ${JSON.stringify(data)};
  144. window.generatedDatetime = ${JSON.stringify(
  145. String(Date())
  146. )};
  147. window.metricsToShow = ${JSON.stringify(
  148. this.metricsToShow
  149. )};
  150. </script>
  151. <script src="bundle.js"></script>
  152. </body>
  153. </html>`
  154. );
  155. cw.close();
  156. }
  157. }
  158. module.exports = HtmlSpaReport;