template.js 4.2 KB

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