network.js 16 KB

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