content-fetch-resources.js 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /*
  2. * Copyright 2010-2019 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * The code in this file is free software: you can redistribute it and/or
  8. * modify it under the terms of the GNU Affero General Public License
  9. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  10. * of the License, or (at your option) any later version.
  11. *
  12. * The code in this file is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  15. * General Public License for more details.
  16. *
  17. * As additional permission under GNU AGPL version 3 section 7, you may
  18. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  19. * AGPL normally required by section 4, provided you include this license
  20. * notice and a URL through which recipients can access the Corresponding
  21. * Source.
  22. */
  23. /* global window */
  24. this.singlefile.lib.fetch.content.resources = this.singlefile.lib.fetch.content.resources || (() => {
  25. const FETCH_REQUEST_EVENT = "single-file-request-fetch";
  26. const FETCH_RESPONSE_EVENT = "single-file-response-fetch";
  27. const browser = this.browser;
  28. const addEventListener = window.addEventListener;
  29. const fetch = window.fetch;
  30. const CustomEvent = window.CustomEvent;
  31. const dispatchEvent = window.dispatchEvent;
  32. const removeEventListener = window.removeEventListener;
  33. return {
  34. fetch: async url => {
  35. try {
  36. let response = await fetch(url, { cache: "force-cache" });
  37. if (response.status == 403) {
  38. response = hostFetch(url);
  39. }
  40. return response;
  41. }
  42. catch (error) {
  43. const responseFetch = await sendMessage({ method: "fetch", url });
  44. return {
  45. status: responseFetch.status,
  46. headers: { get: headerName => responseFetch.headers[headerName] },
  47. arrayBuffer: async () => {
  48. const response = await sendMessage({ method: "fetch.array", requestId: responseFetch.responseId });
  49. return new Uint8Array(response.array).buffer;
  50. }
  51. };
  52. }
  53. }
  54. };
  55. async function sendMessage(message) {
  56. const response = await browser.runtime.sendMessage(message);
  57. if (!response || response.error) {
  58. throw new Error(response && response.error.toString());
  59. } else {
  60. return response;
  61. }
  62. }
  63. function hostFetch(url) {
  64. return new Promise((resolve, reject) => {
  65. dispatchEvent.call(window, new CustomEvent(FETCH_REQUEST_EVENT, { detail: url }));
  66. addEventListener.call(window, FETCH_RESPONSE_EVENT, onResponseFetch, false);
  67. function onResponseFetch(event) {
  68. if (event.detail) {
  69. if (event.detail.url == url) {
  70. removeEventListener.call(window, FETCH_RESPONSE_EVENT, onResponseFetch, false);
  71. if (event.detail.response) {
  72. resolve({
  73. status: event.detail.status,
  74. headers: {
  75. get: name => {
  76. const header = event.detail.headers.find(header => header[0] == name);
  77. return header && header[1];
  78. }
  79. },
  80. arrayBuffer: async () => event.detail.response
  81. });
  82. } else {
  83. reject(event.detail.error);
  84. }
  85. }
  86. } else {
  87. reject();
  88. }
  89. }
  90. });
  91. }
  92. })();