template.js 4.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. "use strict";
  2. /* eslint-disable max-len */
  3. const path = require('path');
  4. const fs = require('fs');
  5. const {
  6. escape
  7. } = require('html-escaper');
  8. const projectRoot = path.resolve(__dirname, '..');
  9. const assetsRoot = path.join(projectRoot, 'public');
  10. exports.renderViewer = renderViewer;
  11. /**
  12. * Escapes `<` characters in JSON to safely use it in `<script>` tag.
  13. */
  14. function escapeJson(json) {
  15. return JSON.stringify(json).replace(/</gu, '\\u003c');
  16. }
  17. function getAssetContent(filename) {
  18. const assetPath = path.join(assetsRoot, filename);
  19. if (!assetPath.startsWith(assetsRoot)) {
  20. throw new Error(`"${filename}" is outside of the assets root`);
  21. }
  22. return fs.readFileSync(assetPath, 'utf8');
  23. }
  24. function html(strings, ...values) {
  25. return strings.map((string, index) => `${string}${values[index] || ''}`).join('');
  26. }
  27. function getScript(filename, mode) {
  28. if (mode === 'static') {
  29. return `<!-- ${escape(filename)} -->
  30. <script>${getAssetContent(filename)}</script>`;
  31. } else {
  32. return `<script src="${escape(filename)}"></script>`;
  33. }
  34. }
  35. function renderViewer({
  36. title,
  37. enableWebSocket,
  38. chartData,
  39. entrypoints,
  40. defaultSizes,
  41. mode
  42. } = {}) {
  43. return html`<!DOCTYPE html>
  44. <html>
  45. <head>
  46. <meta charset="UTF-8"/>
  47. <meta name="viewport" content="width=device-width, initial-scale=1"/>
  48. <title>${escape(title)}</title>
  49. <link rel="shortcut icon" href="" type="image/x-icon" />
  50. <script>
  51. window.enableWebSocket = ${escapeJson(enableWebSocket)};
  52. </script>
  53. ${getScript('viewer.js', mode)}
  54. </head>
  55. <body>
  56. <div id="app"></div>
  57. <script>
  58. window.chartData = ${escapeJson(chartData)};
  59. window.entrypoints = ${escapeJson(entrypoints)};
  60. window.defaultSizes = ${escapeJson(defaultSizes)};
  61. </script>
  62. </body>
  63. </html>`;
  64. }