content-fetch-resources.js 3.0 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  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 = 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: {
  69. get: name => {
  70. const header = event.detail.headers.find(header => header[0] == name);
  71. return header && header[1];
  72. }
  73. },
  74. arrayBuffer: async () => event.detail.response
  75. });
  76. } else {
  77. reject(event.detail.error);
  78. }
  79. }
  80. } else {
  81. reject();
  82. }
  83. }
  84. });
  85. }
  86. })();