content-fetch-resources.js 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  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 browser, fetch, CustomEvent, dispatchEvent, addEventListener, removeEventListener */
  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. return {
  28. fetch: async (url) => {
  29. try {
  30. let response = await fetch(url, { cache: "force-cache" });
  31. if (response.status == 403) {
  32. response = await hostFetch(url);
  33. }
  34. return response;
  35. }
  36. catch (error) {
  37. const responseFetch = await sendMessage({ method: "fetch", url });
  38. return {
  39. status: responseFetch.status,
  40. headers: { get: headerName => responseFetch.headers[headerName] },
  41. arrayBuffer: async () => {
  42. const response = await sendMessage({ method: "fetch.array", requestId: responseFetch.responseId });
  43. return new Uint8Array(response.array).buffer;
  44. }
  45. };
  46. }
  47. }
  48. };
  49. async function sendMessage(message) {
  50. const response = await browser.runtime.sendMessage(message);
  51. if (!response || response.error) {
  52. throw new Error(response && response.error.toString());
  53. } else {
  54. return response;
  55. }
  56. }
  57. function hostFetch(url) {
  58. return new Promise((resolve, reject) => {
  59. dispatchEvent(new CustomEvent(FETCH_REQUEST_EVENT, { detail: url }));
  60. addEventListener(FETCH_RESPONSE_EVENT, onResponseFetch, false);
  61. function onResponseFetch(event) {
  62. if (event.detail) {
  63. if (event.detail.url == url) {
  64. removeEventListener(FETCH_RESPONSE_EVENT, onResponseFetch, false);
  65. if (event.detail.response) {
  66. resolve({
  67. status: event.detail.status,
  68. headers: event.detail.headers,
  69. arrayBuffer: () => new Promise(resolve => resolve(event.detail.response))
  70. });
  71. } else {
  72. reject(event.detail.error);
  73. }
  74. }
  75. } else {
  76. reject();
  77. }
  78. }
  79. });
  80. }
  81. })();