2
0

network.js 16 KB

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