singlefile_companion.js 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  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 path = require("path");
  27. const nativeMessage = require("./lib/messaging.js");
  28. const backEnds = {
  29. jsdom: "./../cli/back-ends/jsdom.js",
  30. puppeteer: "./../cli/back-ends/puppeteer.js",
  31. "puppeteer-firefox": "./../cli/back-ends/puppeteer-firefox.js",
  32. "webdriver-chromium": "./../cli/back-ends/webdriver-chromium.js",
  33. "webdriver-gecko": "./../cli/back-ends/webdriver-gecko.js"
  34. };
  35. process.stdin
  36. .pipe(new nativeMessage.Input())
  37. .pipe(new nativeMessage.Transform(async function (message, push, done) {
  38. try {
  39. await capturePage(message);
  40. push({});
  41. } catch (error) {
  42. push({ error });
  43. }
  44. done();
  45. process.exit(0);
  46. }))
  47. .pipe(new nativeMessage.Output())
  48. .pipe(process.stdout);
  49. async function capturePage(options) {
  50. const companionOptions = require("./options.json");
  51. const backend = require(backEnds[companionOptions.backEnd || "puppeteer"]);
  52. await backend.initialize(companionOptions);
  53. try {
  54. const pageData = await backend.getPageData(options);
  55. pageData.filename = path.resolve("../../", (companionOptions.savePath || ""), pageData.filename);
  56. fs.writeFileSync(getFilename(pageData.filename), pageData.content);
  57. return pageData;
  58. } catch (error) {
  59. if (companionOptions.errorFile) {
  60. const message = "URL: " + options.url + "\nStack: " + error.stack + "\n";
  61. fs.writeFileSync(options.errorFile, message, { flag: "a" });
  62. }
  63. throw error;
  64. } finally {
  65. await backend.closeBrowser();
  66. }
  67. }
  68. function getFilename(filename, index = 1) {
  69. let newFilename = filename;
  70. if (index > 1) {
  71. const regExpMatchExtension = /(\.[^.]+)$/;
  72. const matchExtension = newFilename.match(regExpMatchExtension);
  73. if (matchExtension && matchExtension[1]) {
  74. newFilename = newFilename.replace(regExpMatchExtension, " - " + index + matchExtension[1]);
  75. } else {
  76. newFilename += " - " + index;
  77. }
  78. }
  79. if (fs.existsSync(newFilename)) {
  80. return getFilename(filename, index + 1);
  81. } else {
  82. return newFilename;
  83. }
  84. }