2
0

network.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  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(options) {
  211. this._options = options;
  212. var source = options.source;
  213. this._manager = new NetworkManager(source.url, {
  214. httpHeaders: source.httpHeaders,
  215. withCredentials: source.withCredentials
  216. });
  217. this._rangeChunkSize = source.rangeChunkSize;
  218. this._fullRequestReader = null;
  219. this._rangeRequestReaders = [];
  220. }
  221. PDFNetworkStream.prototype = {
  222. _onRangeRequestReaderClosed: function PDFNetworkStream_onRangeRequestReaderClosed(reader) {
  223. var i = this._rangeRequestReaders.indexOf(reader);
  224. if (i >= 0) {
  225. this._rangeRequestReaders.splice(i, 1);
  226. }
  227. },
  228. getFullReader: function PDFNetworkStream_getFullReader() {
  229. (0, _util.assert)(!this._fullRequestReader);
  230. this._fullRequestReader = new PDFNetworkStreamFullRequestReader(this._manager, this._options);
  231. return this._fullRequestReader;
  232. },
  233. getRangeReader: function PDFNetworkStream_getRangeReader(begin, end) {
  234. var reader = new PDFNetworkStreamRangeRequestReader(this._manager, begin, end);
  235. reader.onClosed = this._onRangeRequestReaderClosed.bind(this);
  236. this._rangeRequestReaders.push(reader);
  237. return reader;
  238. },
  239. cancelAllRequests: function PDFNetworkStream_cancelAllRequests(reason) {
  240. if (this._fullRequestReader) {
  241. this._fullRequestReader.cancel(reason);
  242. }
  243. var readers = this._rangeRequestReaders.slice(0);
  244. readers.forEach(function (reader) {
  245. reader.cancel(reason);
  246. });
  247. }
  248. };
  249. function PDFNetworkStreamFullRequestReader(manager, options) {
  250. this._manager = manager;
  251. var source = options.source;
  252. var args = {
  253. onHeadersReceived: this._onHeadersReceived.bind(this),
  254. onProgressiveData: source.disableStream ? null : this._onProgressiveData.bind(this),
  255. onDone: this._onDone.bind(this),
  256. onError: this._onError.bind(this),
  257. onProgress: this._onProgress.bind(this)
  258. };
  259. this._url = source.url;
  260. this._fullRequestId = manager.requestFull(args);
  261. this._headersReceivedCapability = (0, _util.createPromiseCapability)();
  262. this._disableRange = options.disableRange || false;
  263. this._contentLength = source.length;
  264. this._rangeChunkSize = source.rangeChunkSize;
  265. if (!this._rangeChunkSize && !this._disableRange) {
  266. this._disableRange = true;
  267. }
  268. this._isStreamingSupported = false;
  269. this._isRangeSupported = false;
  270. this._cachedChunks = [];
  271. this._requests = [];
  272. this._done = false;
  273. this._storedError = undefined;
  274. this.onProgress = null;
  275. }
  276. PDFNetworkStreamFullRequestReader.prototype = {
  277. _onHeadersReceived: function PDFNetworkStreamFullRequestReader_onHeadersReceived() {
  278. var fullRequestXhrId = this._fullRequestId;
  279. var fullRequestXhr = this._manager.getRequestXhr(fullRequestXhrId);
  280. var _validateRangeRequest = (0, _network_utils.validateRangeRequestCapabilities)({
  281. getResponseHeader: function getResponseHeader(name) {
  282. return fullRequestXhr.getResponseHeader(name);
  283. },
  284. isHttp: this._manager.isHttp,
  285. rangeChunkSize: this._rangeChunkSize,
  286. disableRange: this._disableRange
  287. }),
  288. allowRangeRequests = _validateRangeRequest.allowRangeRequests,
  289. suggestedLength = _validateRangeRequest.suggestedLength;
  290. this._contentLength = suggestedLength || this._contentLength;
  291. if (allowRangeRequests) {
  292. this._isRangeSupported = true;
  293. }
  294. var networkManager = this._manager;
  295. if (networkManager.isStreamingRequest(fullRequestXhrId)) {
  296. this._isStreamingSupported = true;
  297. } else if (this._isRangeSupported) {
  298. networkManager.abortRequest(fullRequestXhrId);
  299. }
  300. this._headersReceivedCapability.resolve();
  301. },
  302. _onProgressiveData: function PDFNetworkStreamFullRequestReader_onProgressiveData(chunk) {
  303. if (this._requests.length > 0) {
  304. var requestCapability = this._requests.shift();
  305. requestCapability.resolve({
  306. value: chunk,
  307. done: false
  308. });
  309. } else {
  310. this._cachedChunks.push(chunk);
  311. }
  312. },
  313. _onDone: function PDFNetworkStreamFullRequestReader_onDone(args) {
  314. if (args) {
  315. this._onProgressiveData(args.chunk);
  316. }
  317. this._done = true;
  318. if (this._cachedChunks.length > 0) {
  319. return;
  320. }
  321. this._requests.forEach(function (requestCapability) {
  322. requestCapability.resolve({
  323. value: undefined,
  324. done: true
  325. });
  326. });
  327. this._requests = [];
  328. },
  329. _onError: function PDFNetworkStreamFullRequestReader_onError(status) {
  330. var url = this._url;
  331. var exception = (0, _network_utils.createResponseStatusError)(status, url);
  332. this._storedError = exception;
  333. this._headersReceivedCapability.reject(exception);
  334. this._requests.forEach(function (requestCapability) {
  335. requestCapability.reject(exception);
  336. });
  337. this._requests = [];
  338. this._cachedChunks = [];
  339. },
  340. _onProgress: function PDFNetworkStreamFullRequestReader_onProgress(data) {
  341. if (this.onProgress) {
  342. this.onProgress({
  343. loaded: data.loaded,
  344. total: data.lengthComputable ? data.total : this._contentLength
  345. });
  346. }
  347. },
  348. get isRangeSupported() {
  349. return this._isRangeSupported;
  350. },
  351. get isStreamingSupported() {
  352. return this._isStreamingSupported;
  353. },
  354. get contentLength() {
  355. return this._contentLength;
  356. },
  357. get headersReady() {
  358. return this._headersReceivedCapability.promise;
  359. },
  360. read: function PDFNetworkStreamFullRequestReader_read() {
  361. if (this._storedError) {
  362. return Promise.reject(this._storedError);
  363. }
  364. if (this._cachedChunks.length > 0) {
  365. var chunk = this._cachedChunks.shift();
  366. return Promise.resolve({
  367. value: chunk,
  368. done: false
  369. });
  370. }
  371. if (this._done) {
  372. return Promise.resolve({
  373. value: undefined,
  374. done: true
  375. });
  376. }
  377. var requestCapability = (0, _util.createPromiseCapability)();
  378. this._requests.push(requestCapability);
  379. return requestCapability.promise;
  380. },
  381. cancel: function PDFNetworkStreamFullRequestReader_cancel(reason) {
  382. this._done = true;
  383. this._headersReceivedCapability.reject(reason);
  384. this._requests.forEach(function (requestCapability) {
  385. requestCapability.resolve({
  386. value: undefined,
  387. done: true
  388. });
  389. });
  390. this._requests = [];
  391. if (this._manager.isPendingRequest(this._fullRequestId)) {
  392. this._manager.abortRequest(this._fullRequestId);
  393. }
  394. this._fullRequestReader = null;
  395. }
  396. };
  397. function PDFNetworkStreamRangeRequestReader(manager, begin, end) {
  398. this._manager = manager;
  399. var args = {
  400. onDone: this._onDone.bind(this),
  401. onProgress: this._onProgress.bind(this)
  402. };
  403. this._requestId = manager.requestRange(begin, end, args);
  404. this._requests = [];
  405. this._queuedChunk = null;
  406. this._done = false;
  407. this.onProgress = null;
  408. this.onClosed = null;
  409. }
  410. PDFNetworkStreamRangeRequestReader.prototype = {
  411. _close: function PDFNetworkStreamRangeRequestReader_close() {
  412. if (this.onClosed) {
  413. this.onClosed(this);
  414. }
  415. },
  416. _onDone: function PDFNetworkStreamRangeRequestReader_onDone(data) {
  417. var chunk = data.chunk;
  418. if (this._requests.length > 0) {
  419. var requestCapability = this._requests.shift();
  420. requestCapability.resolve({
  421. value: chunk,
  422. done: false
  423. });
  424. } else {
  425. this._queuedChunk = chunk;
  426. }
  427. this._done = true;
  428. this._requests.forEach(function (requestCapability) {
  429. requestCapability.resolve({
  430. value: undefined,
  431. done: true
  432. });
  433. });
  434. this._requests = [];
  435. this._close();
  436. },
  437. _onProgress: function PDFNetworkStreamRangeRequestReader_onProgress(evt) {
  438. if (!this.isStreamingSupported && this.onProgress) {
  439. this.onProgress({ loaded: evt.loaded });
  440. }
  441. },
  442. get isStreamingSupported() {
  443. return false;
  444. },
  445. read: function PDFNetworkStreamRangeRequestReader_read() {
  446. if (this._queuedChunk !== null) {
  447. var chunk = this._queuedChunk;
  448. this._queuedChunk = null;
  449. return Promise.resolve({
  450. value: chunk,
  451. done: false
  452. });
  453. }
  454. if (this._done) {
  455. return Promise.resolve({
  456. value: undefined,
  457. done: true
  458. });
  459. }
  460. var requestCapability = (0, _util.createPromiseCapability)();
  461. this._requests.push(requestCapability);
  462. return requestCapability.promise;
  463. },
  464. cancel: function PDFNetworkStreamRangeRequestReader_cancel(reason) {
  465. this._done = true;
  466. this._requests.forEach(function (requestCapability) {
  467. requestCapability.resolve({
  468. value: undefined,
  469. done: true
  470. });
  471. });
  472. this._requests = [];
  473. if (this._manager.isPendingRequest(this._requestId)) {
  474. this._manager.abortRequest(this._requestId);
  475. }
  476. this._close();
  477. }
  478. };
  479. exports.PDFNetworkStream = PDFNetworkStream;
  480. exports.NetworkManager = NetworkManager;