singlefile_companion.js 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. #!/usr/local/bin/node
  2. /*
  3. * Copyright 2010-2020 Gildas Lormeau
  4. * contact : gildas.lormeau <at> gmail.com
  5. *
  6. * This file is part of SingleFile.
  7. *
  8. * The code in this file is free software: you can redistribute it and/or
  9. * modify it under the terms of the GNU Affero General Public License
  10. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  11. * of the License, or (at your option) any later version.
  12. *
  13. * The code in this file is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  16. * General Public License for more details.
  17. *
  18. * As additional permission under GNU AGPL version 3 section 7, you may
  19. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  20. * AGPL normally required by section 4, provided you include this license
  21. * notice and a URL through which recipients can access the Corresponding
  22. * Source.
  23. */
  24. /* global require, process */
  25. const fs = require("fs");
  26. const nativeMessage = require("./lib/messaging.js");
  27. const backEnds = {
  28. jsdom: "./../cli/back-ends/jsdom.js",
  29. puppeteer: "./../cli/back-ends/puppeteer.js",
  30. "puppeteer-firefox": "./../cli/back-ends/puppeteer-firefox.js",
  31. "webdriver-chromium": "./../cli/back-ends/webdriver-chromium.js",
  32. "webdriver-gecko": "./../cli/back-ends/webdriver-gecko.js"
  33. };
  34. process.stdin
  35. .pipe(new nativeMessage.Input())
  36. .pipe(new nativeMessage.Transform(function (msg, push, done) {
  37. capturePage(msg);
  38. push(JSON.stringify(msg));
  39. done();
  40. }))
  41. .pipe(new nativeMessage.Output())
  42. .pipe(process.stdout);
  43. async function capturePage(options) {
  44. const companionOptions = require("./options.json");
  45. const backend = require(backEnds[companionOptions.backEnd || "puppeteer"]);
  46. await backend.initialize(companionOptions);
  47. try {
  48. const pageData = await backend.getPageData(options);
  49. pageData.filename = "../../" + pageData.filename;
  50. fs.writeFileSync(getFilename(pageData.filename), pageData.content);
  51. return pageData;
  52. } catch (error) {
  53. if (companionOptions.errorFile) {
  54. const message = "URL: " + options.url + "\nStack: " + error.stack + "\n";
  55. fs.writeFileSync(options.errorFile, message, { flag: "a" });
  56. }
  57. } finally {
  58. await backend.closeBrowser();
  59. process.exit(0);
  60. }
  61. }
  62. function getFilename(filename, index = 1) {
  63. let newFilename = filename;
  64. if (index > 1) {
  65. const regExpMatchExtension = /(\.[^.]+)$/;
  66. const matchExtension = newFilename.match(regExpMatchExtension);
  67. if (matchExtension && matchExtension[1]) {
  68. newFilename = newFilename.replace(regExpMatchExtension, " - " + index + matchExtension[1]);
  69. } else {
  70. newFilename += " - " + index;
  71. }
  72. }
  73. if (fs.existsSync(newFilename)) {
  74. return getFilename(filename, index + 1);
  75. } else {
  76. return newFilename;
  77. }
  78. }