2
0

network.js 16 KB

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