render_cover.js 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. #!/usr/bin/env node
  2. /**
  3. * render_cover.js — Render cover.html → cover.pdf via Playwright.
  4. *
  5. * Usage:
  6. * node render_cover.js --input cover.html --out cover.pdf
  7. * node render_cover.js --input cover.html --out cover.pdf --wait 1200
  8. *
  9. * Exit codes: 0 success, 1 bad args, 2 dependency missing, 3 render error
  10. */
  11. const path = require("path");
  12. const fs = require("fs");
  13. function usage() {
  14. console.error("Usage: node render_cover.js --input <file.html> --out <file.pdf> [--wait <ms>]");
  15. process.exit(1);
  16. }
  17. // ── Arg parsing ────────────────────────────────────────────────────────────────
  18. const args = process.argv.slice(2);
  19. let inputFile = null, outFile = null, waitMs = 800;
  20. for (let i = 0; i < args.length; i++) {
  21. if (args[i] === "--input" && args[i + 1]) { inputFile = args[++i]; }
  22. else if (args[i] === "--out" && args[i + 1]) { outFile = args[++i]; }
  23. else if (args[i] === "--wait" && args[i + 1]) { waitMs = parseInt(args[++i], 10); }
  24. }
  25. if (!inputFile || !outFile) usage();
  26. if (!fs.existsSync(inputFile)) {
  27. console.error(JSON.stringify({ status: "error", error: `File not found: ${inputFile}` }));
  28. process.exit(1);
  29. }
  30. // ── Playwright loader (tolerates global npm installs) ─────────────────────────
  31. function loadPlaywright() {
  32. const { execSync } = require("child_process");
  33. try { return require("playwright"); } catch (_) {}
  34. try {
  35. const root = execSync("npm root -g", { stdio: ["ignore","pipe","ignore"] }).toString().trim();
  36. return require(path.join(root, "playwright"));
  37. } catch (_) {}
  38. console.error(JSON.stringify({
  39. status: "error",
  40. error: "playwright not found",
  41. hint: "Run: npm install -g playwright && npx playwright install chromium"
  42. }));
  43. process.exit(2);
  44. }
  45. // ── Main ───────────────────────────────────────────────────────────────────────
  46. (async () => {
  47. const { chromium } = loadPlaywright();
  48. let browser;
  49. try {
  50. browser = await chromium.launch();
  51. } catch (e) {
  52. // Chromium binary missing — try installing
  53. const { spawnSync } = require("child_process");
  54. const r = spawnSync("npx", ["playwright", "install", "chromium"], { stdio: "inherit", shell: true });
  55. if (r.status !== 0) {
  56. console.error(JSON.stringify({
  57. status: "error",
  58. error: "Chromium not installed and auto-install failed",
  59. hint: "Run: npx playwright install chromium"
  60. }));
  61. process.exit(2);
  62. }
  63. browser = await chromium.launch();
  64. }
  65. try {
  66. const page = await browser.newPage();
  67. const fileUrl = "file://" + path.resolve(inputFile);
  68. await page.goto(fileUrl);
  69. await page.waitForTimeout(waitMs); // let CSS + any JS settle
  70. await page.pdf({
  71. path: outFile,
  72. width: "794px",
  73. height: "1123px",
  74. printBackground: true,
  75. });
  76. await browser.close();
  77. // Basic sanity: output file must exist and be > 5 KB
  78. const stat = fs.statSync(outFile);
  79. if (stat.size < 5000) {
  80. console.error(JSON.stringify({
  81. status: "error",
  82. error: "Output PDF is suspiciously small — cover may be blank",
  83. hint: "Check cover.html for render errors"
  84. }));
  85. process.exit(3);
  86. }
  87. console.log(JSON.stringify({
  88. status: "ok",
  89. out: outFile,
  90. size_kb: Math.round(stat.size / 1024),
  91. }));
  92. } catch (e) {
  93. if (browser) await browser.close().catch(() => {});
  94. console.error(JSON.stringify({ status: "error", error: String(e) }));
  95. process.exit(3);
  96. }
  97. })();