network.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487
  1. /* Copyright 2017 Mozilla Foundation
  2. *
  3. * Licensed under the Apache License, Version 2.0 (the "License");
  4. * you may not use this file except in compliance with the License.
  5. * You may obtain a copy of the License at
  6. *
  7. * http://www.apache.org/licenses/LICENSE-2.0
  8. *
  9. * Unless required by applicable law or agreed to in writing, software
  10. * distributed under the License is distributed on an "AS IS" BASIS,
  11. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. * See the License for the specific language governing permissions and
  13. * limitations under the License.
  14. */
  15. 'use strict';
  16. Object.defineProperty(exports, "__esModule", {
  17. value: true
  18. });
  19. exports.NetworkManager = exports.PDFNetworkStream = undefined;
  20. var _util = require('../shared/util');
  21. var _network_utils = require('./network_utils');
  22. var _global_scope = require('../shared/global_scope');
  23. var _global_scope2 = _interopRequireDefault(_global_scope);
  24. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  25. ;
  26. var OK_RESPONSE = 200;
  27. var PARTIAL_CONTENT_RESPONSE = 206;
  28. function NetworkManager(url, args) {
  29. this.url = url;
  30. args = args || {};
  31. this.isHttp = /^https?:/i.test(url);
  32. this.httpHeaders = this.isHttp && args.httpHeaders || {};
  33. this.withCredentials = args.withCredentials || false;
  34. this.getXhr = args.getXhr || function NetworkManager_getXhr() {
  35. return new XMLHttpRequest();
  36. };
  37. this.currXhrId = 0;
  38. this.pendingRequests = Object.create(null);
  39. this.loadedRequests = Object.create(null);
  40. }
  41. function getArrayBuffer(xhr) {
  42. var data = xhr.response;
  43. if (typeof data !== 'string') {
  44. return data;
  45. }
  46. var array = (0, _util.stringToBytes)(data);
  47. return array.buffer;
  48. }
  49. var supportsMozChunked = function supportsMozChunkedClosure() {
  50. try {
  51. var x = new XMLHttpRequest();
  52. x.open('GET', _global_scope2.default.location.href);
  53. x.responseType = 'moz-chunked-arraybuffer';
  54. return x.responseType === 'moz-chunked-arraybuffer';
  55. } catch (e) {
  56. return false;
  57. }
  58. }();
  59. NetworkManager.prototype = {
  60. requestRange: function NetworkManager_requestRange(begin, end, listeners) {
  61. var args = {
  62. begin: begin,
  63. end: end
  64. };
  65. for (var prop in listeners) {
  66. args[prop] = listeners[prop];
  67. }
  68. return this.request(args);
  69. },
  70. requestFull: function NetworkManager_requestFull(listeners) {
  71. return this.request(listeners);
  72. },
  73. request: function NetworkManager_request(args) {
  74. var xhr = this.getXhr();
  75. var xhrId = this.currXhrId++;
  76. var pendingRequest = this.pendingRequests[xhrId] = { xhr: xhr };
  77. xhr.open('GET', this.url);
  78. xhr.withCredentials = this.withCredentials;
  79. for (var property in this.httpHeaders) {
  80. var value = this.httpHeaders[property];
  81. if (typeof value === 'undefined') {
  82. continue;
  83. }
  84. xhr.setRequestHeader(property, value);
  85. }
  86. if (this.isHttp && 'begin' in args && 'end' in args) {
  87. var rangeStr = args.begin + '-' + (args.end - 1);
  88. xhr.setRequestHeader('Range', 'bytes=' + rangeStr);
  89. pendingRequest.expectedStatus = 206;
  90. } else {
  91. pendingRequest.expectedStatus = 200;
  92. }
  93. var useMozChunkedLoading = supportsMozChunked && !!args.onProgressiveData;
  94. if (useMozChunkedLoading) {
  95. xhr.responseType = 'moz-chunked-arraybuffer';
  96. pendingRequest.onProgressiveData = args.onProgressiveData;
  97. pendingRequest.mozChunked = true;
  98. } else {
  99. xhr.responseType = 'arraybuffer';
  100. }
  101. if (args.onError) {
  102. xhr.onerror = function (evt) {
  103. args.onError(xhr.status);
  104. };
  105. }
  106. xhr.onreadystatechange = this.onStateChange.bind(this, xhrId);
  107. xhr.onprogress = this.onProgress.bind(this, xhrId);
  108. pendingRequest.onHeadersReceived = args.onHeadersReceived;
  109. pendingRequest.onDone = args.onDone;
  110. pendingRequest.onError = args.onError;
  111. pendingRequest.onProgress = args.onProgress;
  112. xhr.send(null);
  113. return xhrId;
  114. },
  115. onProgress: function NetworkManager_onProgress(xhrId, evt) {
  116. var pendingRequest = this.pendingRequests[xhrId];
  117. if (!pendingRequest) {
  118. return;
  119. }
  120. if (pendingRequest.mozChunked) {
  121. var chunk = getArrayBuffer(pendingRequest.xhr);
  122. pendingRequest.onProgressiveData(chunk);
  123. }
  124. var onProgress = pendingRequest.onProgress;
  125. if (onProgress) {
  126. onProgress(evt);
  127. }
  128. },
  129. onStateChange: function NetworkManager_onStateChange(xhrId, evt) {
  130. var pendingRequest = this.pendingRequests[xhrId];
  131. if (!pendingRequest) {
  132. return;
  133. }
  134. var xhr = pendingRequest.xhr;
  135. if (xhr.readyState >= 2 && pendingRequest.onHeadersReceived) {
  136. pendingRequest.onHeadersReceived();
  137. delete pendingRequest.onHeadersReceived;
  138. }
  139. if (xhr.readyState !== 4) {
  140. return;
  141. }
  142. if (!(xhrId in this.pendingRequests)) {
  143. return;
  144. }
  145. delete this.pendingRequests[xhrId];
  146. if (xhr.status === 0 && this.isHttp) {
  147. if (pendingRequest.onError) {
  148. pendingRequest.onError(xhr.status);
  149. }
  150. return;
  151. }
  152. var xhrStatus = xhr.status || OK_RESPONSE;
  153. var ok_response_on_range_request = xhrStatus === OK_RESPONSE && pendingRequest.expectedStatus === PARTIAL_CONTENT_RESPONSE;
  154. if (!ok_response_on_range_request && xhrStatus !== pendingRequest.expectedStatus) {
  155. if (pendingRequest.onError) {
  156. pendingRequest.onError(xhr.status);
  157. }
  158. return;
  159. }
  160. this.loadedRequests[xhrId] = true;
  161. var chunk = getArrayBuffer(xhr);
  162. if (xhrStatus === PARTIAL_CONTENT_RESPONSE) {
  163. var rangeHeader = xhr.getResponseHeader('Content-Range');
  164. var matches = /bytes (\d+)-(\d+)\/(\d+)/.exec(rangeHeader);
  165. var begin = parseInt(matches[1], 10);
  166. pendingRequest.onDone({
  167. begin: begin,
  168. chunk: chunk
  169. });
  170. } else if (pendingRequest.onProgressiveData) {
  171. pendingRequest.onDone(null);
  172. } else if (chunk) {
  173. pendingRequest.onDone({
  174. begin: 0,
  175. chunk: chunk
  176. });
  177. } else if (pendingRequest.onError) {
  178. pendingRequest.onError(xhr.status);
  179. }
  180. },
  181. hasPendingRequests: function NetworkManager_hasPendingRequests() {
  182. for (var xhrId in this.pendingRequests) {
  183. return true;
  184. }
  185. return false;
  186. },
  187. getRequestXhr: function NetworkManager_getXhr(xhrId) {
  188. return this.pendingRequests[xhrId].xhr;
  189. },
  190. isStreamingRequest: function NetworkManager_isStreamingRequest(xhrId) {
  191. return !!this.pendingRequests[xhrId].onProgressiveData;
  192. },
  193. isPendingRequest: function NetworkManager_isPendingRequest(xhrId) {
  194. return xhrId in this.pendingRequests;
  195. },
  196. isLoadedRequest: function NetworkManager_isLoadedRequest(xhrId) {
  197. return xhrId in this.loadedRequests;
  198. },
  199. abortAllRequests: function NetworkManager_abortAllRequests() {
  200. for (var xhrId in this.pendingRequests) {
  201. this.abortRequest(xhrId | 0);
  202. }
  203. },
  204. abortRequest: function NetworkManager_abortRequest(xhrId) {
  205. var xhr = this.pendingRequests[xhrId].xhr;
  206. delete this.pendingRequests[xhrId];
  207. xhr.abort();
  208. }
  209. };
  210. function PDFNetworkStream(source) {
  211. this._source = source;
  212. this._manager = new NetworkManager(source.url, {
  213. httpHeaders: source.httpHeaders,
  214. withCredentials: source.withCredentials
  215. });
  216. this._rangeChunkSize = source.rangeChunkSize;
  217. this._fullRequestReader = null;
  218. this._rangeRequestReaders = [];
  219. }
  220. PDFNetworkStream.prototype = {
  221. _onRangeRequestReaderClosed: function PDFNetworkStream_onRangeRequestReaderClosed(reader) {
  222. var i = this._rangeRequestReaders.indexOf(reader);
  223. if (i >= 0) {
  224. this._rangeRequestReaders.splice(i, 1);
  225. }
  226. },
  227. getFullReader: function PDFNetworkStream_getFullReader() {
  228. (0, _util.assert)(!this._fullRequestReader);
  229. this._fullRequestReader = new PDFNetworkStreamFullRequestReader(this._manager, this._source);
  230. return this._fullRequestReader;
  231. },
  232. getRangeReader: function PDFNetworkStream_getRangeReader(begin, end) {
  233. var reader = new PDFNetworkStreamRangeRequestReader(this._manager, begin, end);
  234. reader.onClosed = this._onRangeRequestReaderClosed.bind(this);
  235. this._rangeRequestReaders.push(reader);
  236. return reader;
  237. },
  238. cancelAllRequests: function PDFNetworkStream_cancelAllRequests(reason) {
  239. if (this._fullRequestReader) {
  240. this._fullRequestReader.cancel(reason);
  241. }
  242. var readers = this._rangeRequestReaders.slice(0);
  243. readers.forEach(function (reader) {
  244. reader.cancel(reason);
  245. });
  246. }
  247. };
  248. function PDFNetworkStreamFullRequestReader(manager, source) {
  249. this._manager = manager;
  250. var args = {
  251. onHeadersReceived: this._onHeadersReceived.bind(this),
  252. onProgressiveData: source.disableStream ? null : this._onProgressiveData.bind(this),
  253. onDone: this._onDone.bind(this),
  254. onError: this._onError.bind(this),
  255. onProgress: this._onProgress.bind(this)
  256. };
  257. this._url = source.url;
  258. this._fullRequestId = manager.requestFull(args);
  259. this._headersReceivedCapability = (0, _util.createPromiseCapability)();
  260. this._disableRange = source.disableRange || false;
  261. this._contentLength = source.length;
  262. this._rangeChunkSize = source.rangeChunkSize;
  263. if (!this._rangeChunkSize && !this._disableRange) {
  264. this._disableRange = true;
  265. }
  266. this._isStreamingSupported = false;
  267. this._isRangeSupported = false;
  268. this._cachedChunks = [];
  269. this._requests = [];
  270. this._done = false;
  271. this._storedError = undefined;
  272. this.onProgress = null;
  273. }
  274. PDFNetworkStreamFullRequestReader.prototype = {
  275. _onHeadersReceived: function PDFNetworkStreamFullRequestReader_onHeadersReceived() {
  276. var fullRequestXhrId = this._fullRequestId;
  277. var fullRequestXhr = this._manager.getRequestXhr(fullRequestXhrId);
  278. var _validateRangeRequest = (0, _network_utils.validateRangeRequestCapabilities)({
  279. getResponseHeader: function getResponseHeader(name) {
  280. return fullRequestXhr.getResponseHeader(name);
  281. },
  282. isHttp: this._manager.isHttp,
  283. rangeChunkSize: this._rangeChunkSize,
  284. disableRange: this._disableRange
  285. }),
  286. allowRangeRequests = _validateRangeRequest.allowRangeRequests,
  287. suggestedLength = _validateRangeRequest.suggestedLength;
  288. this._contentLength = suggestedLength || this._contentLength;
  289. if (allowRangeRequests) {
  290. this._isRangeSupported = true;
  291. }
  292. var networkManager = this._manager;
  293. if (networkManager.isStreamingRequest(fullRequestXhrId)) {
  294. this._isStreamingSupported = true;
  295. } else if (this._isRangeSupported) {
  296. networkManager.abortRequest(fullRequestXhrId);
  297. }
  298. this._headersReceivedCapability.resolve();
  299. },
  300. _onProgressiveData: function PDFNetworkStreamFullRequestReader_onProgressiveData(chunk) {
  301. if (this._requests.length > 0) {
  302. var requestCapability = this._requests.shift();
  303. requestCapability.resolve({
  304. value: chunk,
  305. done: false
  306. });
  307. } else {
  308. this._cachedChunks.push(chunk);
  309. }
  310. },
  311. _onDone: function PDFNetworkStreamFullRequestReader_onDone(args) {
  312. if (args) {
  313. this._onProgressiveData(args.chunk);
  314. }
  315. this._done = true;
  316. if (this._cachedChunks.length > 0) {
  317. return;
  318. }
  319. this._requests.forEach(function (requestCapability) {
  320. requestCapability.resolve({
  321. value: undefined,
  322. done: true
  323. });
  324. });
  325. this._requests = [];
  326. },
  327. _onError: function PDFNetworkStreamFullRequestReader_onError(status) {
  328. var url = this._url;
  329. var exception = (0, _network_utils.createResponseStatusError)(status, url);
  330. this._storedError = exception;
  331. this._headersReceivedCapability.reject(exception);
  332. this._requests.forEach(function (requestCapability) {
  333. requestCapability.reject(exception);
  334. });
  335. this._requests = [];
  336. this._cachedChunks = [];
  337. },
  338. _onProgress: function PDFNetworkStreamFullRequestReader_onProgress(data) {
  339. if (this.onProgress) {
  340. this.onProgress({
  341. loaded: data.loaded,
  342. total: data.lengthComputable ? data.total : this._contentLength
  343. });
  344. }
  345. },
  346. get isRangeSupported() {
  347. return this._isRangeSupported;
  348. },
  349. get isStreamingSupported() {
  350. return this._isStreamingSupported;
  351. },
  352. get contentLength() {
  353. return this._contentLength;
  354. },
  355. get headersReady() {
  356. return this._headersReceivedCapability.promise;
  357. },
  358. read: function PDFNetworkStreamFullRequestReader_read() {
  359. if (this._storedError) {
  360. return Promise.reject(this._storedError);
  361. }
  362. if (this._cachedChunks.length > 0) {
  363. var chunk = this._cachedChunks.shift();
  364. return Promise.resolve({
  365. value: chunk,
  366. done: false
  367. });
  368. }
  369. if (this._done) {
  370. return Promise.resolve({
  371. value: undefined,
  372. done: true
  373. });
  374. }
  375. var requestCapability = (0, _util.createPromiseCapability)();
  376. this._requests.push(requestCapability);
  377. return requestCapability.promise;
  378. },
  379. cancel: function PDFNetworkStreamFullRequestReader_cancel(reason) {
  380. this._done = true;
  381. this._headersReceivedCapability.reject(reason);
  382. this._requests.forEach(function (requestCapability) {
  383. requestCapability.resolve({
  384. value: undefined,
  385. done: true
  386. });
  387. });
  388. this._requests = [];
  389. if (this._manager.isPendingRequest(this._fullRequestId)) {
  390. this._manager.abortRequest(this._fullRequestId);
  391. }
  392. this._fullRequestReader = null;
  393. }
  394. };
  395. function PDFNetworkStreamRangeRequestReader(manager, begin, end) {
  396. this._manager = manager;
  397. var args = {
  398. onDone: this._onDone.bind(this),
  399. onProgress: this._onProgress.bind(this)
  400. };
  401. this._requestId = manager.requestRange(begin, end, args);
  402. this._requests = [];
  403. this._queuedChunk = null;
  404. this._done = false;
  405. this.onProgress = null;
  406. this.onClosed = null;
  407. }
  408. PDFNetworkStreamRangeRequestReader.prototype = {
  409. _close: function PDFNetworkStreamRangeRequestReader_close() {
  410. if (this.onClosed) {
  411. this.onClosed(this);
  412. }
  413. },
  414. _onDone: function PDFNetworkStreamRangeRequestReader_onDone(data) {
  415. var chunk = data.chunk;
  416. if (this._requests.length > 0) {
  417. var requestCapability = this._requests.shift();
  418. requestCapability.resolve({
  419. value: chunk,
  420. done: false
  421. });
  422. } else {
  423. this._queuedChunk = chunk;
  424. }
  425. this._done = true;
  426. this._requests.forEach(function (requestCapability) {
  427. requestCapability.resolve({
  428. value: undefined,
  429. done: true
  430. });
  431. });
  432. this._requests = [];
  433. this._close();
  434. },
  435. _onProgress: function PDFNetworkStreamRangeRequestReader_onProgress(evt) {
  436. if (!this.isStreamingSupported && this.onProgress) {
  437. this.onProgress({ loaded: evt.loaded });
  438. }
  439. },
  440. get isStreamingSupported() {
  441. return false;
  442. },
  443. read: function PDFNetworkStreamRangeRequestReader_read() {
  444. if (this._queuedChunk !== null) {
  445. var chunk = this._queuedChunk;
  446. this._queuedChunk = null;
  447. return Promise.resolve({
  448. value: chunk,
  449. done: false
  450. });
  451. }
  452. if (this._done) {
  453. return Promise.resolve({
  454. value: undefined,
  455. done: true
  456. });
  457. }
  458. var requestCapability = (0, _util.createPromiseCapability)();
  459. this._requests.push(requestCapability);
  460. return requestCapability.promise;
  461. },
  462. cancel: function PDFNetworkStreamRangeRequestReader_cancel(reason) {
  463. this._done = true;
  464. this._requests.forEach(function (requestCapability) {
  465. requestCapability.resolve({
  466. value: undefined,
  467. done: true
  468. });
  469. });
  470. this._requests = [];
  471. if (this._manager.isPendingRequest(this._requestId)) {
  472. this._manager.abortRequest(this._requestId);
  473. }
  474. this._close();
  475. }
  476. };
  477. exports.PDFNetworkStream = PDFNetworkStream;
  478. exports.NetworkManager = NetworkManager;