worker.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
  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. var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
  17. var sharedUtil = require('../shared/util.js');
  18. var corePrimitives = require('./primitives.js');
  19. var corePdfManager = require('./pdf_manager.js');
  20. var UNSUPPORTED_FEATURES = sharedUtil.UNSUPPORTED_FEATURES;
  21. var InvalidPDFException = sharedUtil.InvalidPDFException;
  22. var MessageHandler = sharedUtil.MessageHandler;
  23. var MissingPDFException = sharedUtil.MissingPDFException;
  24. var UnexpectedResponseException = sharedUtil.UnexpectedResponseException;
  25. var PasswordException = sharedUtil.PasswordException;
  26. var UnknownErrorException = sharedUtil.UnknownErrorException;
  27. var XRefParseException = sharedUtil.XRefParseException;
  28. var arrayByteLength = sharedUtil.arrayByteLength;
  29. var arraysToBytes = sharedUtil.arraysToBytes;
  30. var assert = sharedUtil.assert;
  31. var createPromiseCapability = sharedUtil.createPromiseCapability;
  32. var info = sharedUtil.info;
  33. var warn = sharedUtil.warn;
  34. var setVerbosityLevel = sharedUtil.setVerbosityLevel;
  35. var isNodeJS = sharedUtil.isNodeJS;
  36. var Ref = corePrimitives.Ref;
  37. var LocalPdfManager = corePdfManager.LocalPdfManager;
  38. var NetworkPdfManager = corePdfManager.NetworkPdfManager;
  39. var WorkerTask = function WorkerTaskClosure() {
  40. function WorkerTask(name) {
  41. this.name = name;
  42. this.terminated = false;
  43. this._capability = createPromiseCapability();
  44. }
  45. WorkerTask.prototype = {
  46. get finished() {
  47. return this._capability.promise;
  48. },
  49. finish: function finish() {
  50. this._capability.resolve();
  51. },
  52. terminate: function terminate() {
  53. this.terminated = true;
  54. },
  55. ensureNotTerminated: function ensureNotTerminated() {
  56. if (this.terminated) {
  57. throw new Error('Worker task was terminated');
  58. }
  59. }
  60. };
  61. return WorkerTask;
  62. }();
  63. var PDFWorkerStream = function PDFWorkerStreamClosure() {
  64. function PDFWorkerStream(params, msgHandler) {
  65. this._queuedChunks = [];
  66. var initialData = params.initialData;
  67. if (initialData && initialData.length > 0) {
  68. this._queuedChunks.push(initialData);
  69. }
  70. this._msgHandler = msgHandler;
  71. this._isRangeSupported = !params.disableRange;
  72. this._isStreamingSupported = !params.disableStream;
  73. this._contentLength = params.length;
  74. this._fullRequestReader = null;
  75. this._rangeReaders = [];
  76. msgHandler.on('OnDataRange', this._onReceiveData.bind(this));
  77. msgHandler.on('OnDataProgress', this._onProgress.bind(this));
  78. }
  79. PDFWorkerStream.prototype = {
  80. _onReceiveData: function PDFWorkerStream_onReceiveData(args) {
  81. if (args.begin === undefined) {
  82. if (this._fullRequestReader) {
  83. this._fullRequestReader._enqueue(args.chunk);
  84. } else {
  85. this._queuedChunks.push(args.chunk);
  86. }
  87. } else {
  88. var found = this._rangeReaders.some(function (rangeReader) {
  89. if (rangeReader._begin !== args.begin) {
  90. return false;
  91. }
  92. rangeReader._enqueue(args.chunk);
  93. return true;
  94. });
  95. assert(found);
  96. }
  97. },
  98. _onProgress: function PDFWorkerStream_onProgress(evt) {
  99. if (this._rangeReaders.length > 0) {
  100. var firstReader = this._rangeReaders[0];
  101. if (firstReader.onProgress) {
  102. firstReader.onProgress({ loaded: evt.loaded });
  103. }
  104. }
  105. },
  106. _removeRangeReader: function PDFWorkerStream_removeRangeReader(reader) {
  107. var i = this._rangeReaders.indexOf(reader);
  108. if (i >= 0) {
  109. this._rangeReaders.splice(i, 1);
  110. }
  111. },
  112. getFullReader: function PDFWorkerStream_getFullReader() {
  113. assert(!this._fullRequestReader);
  114. var queuedChunks = this._queuedChunks;
  115. this._queuedChunks = null;
  116. return new PDFWorkerStreamReader(this, queuedChunks);
  117. },
  118. getRangeReader: function PDFWorkerStream_getRangeReader(begin, end) {
  119. var reader = new PDFWorkerStreamRangeReader(this, begin, end);
  120. this._msgHandler.send('RequestDataRange', {
  121. begin: begin,
  122. end: end
  123. });
  124. this._rangeReaders.push(reader);
  125. return reader;
  126. },
  127. cancelAllRequests: function PDFWorkerStream_cancelAllRequests(reason) {
  128. if (this._fullRequestReader) {
  129. this._fullRequestReader.cancel(reason);
  130. }
  131. var readers = this._rangeReaders.slice(0);
  132. readers.forEach(function (rangeReader) {
  133. rangeReader.cancel(reason);
  134. });
  135. }
  136. };
  137. function PDFWorkerStreamReader(stream, queuedChunks) {
  138. this._stream = stream;
  139. this._done = false;
  140. this._queuedChunks = queuedChunks || [];
  141. this._requests = [];
  142. this._headersReady = Promise.resolve();
  143. stream._fullRequestReader = this;
  144. this.onProgress = null;
  145. }
  146. PDFWorkerStreamReader.prototype = {
  147. _enqueue: function PDFWorkerStreamReader_enqueue(chunk) {
  148. if (this._done) {
  149. return;
  150. }
  151. if (this._requests.length > 0) {
  152. var requestCapability = this._requests.shift();
  153. requestCapability.resolve({
  154. value: chunk,
  155. done: false
  156. });
  157. return;
  158. }
  159. this._queuedChunks.push(chunk);
  160. },
  161. get headersReady() {
  162. return this._headersReady;
  163. },
  164. get isRangeSupported() {
  165. return this._stream._isRangeSupported;
  166. },
  167. get isStreamingSupported() {
  168. return this._stream._isStreamingSupported;
  169. },
  170. get contentLength() {
  171. return this._stream._contentLength;
  172. },
  173. read: function PDFWorkerStreamReader_read() {
  174. if (this._queuedChunks.length > 0) {
  175. var chunk = this._queuedChunks.shift();
  176. return Promise.resolve({
  177. value: chunk,
  178. done: false
  179. });
  180. }
  181. if (this._done) {
  182. return Promise.resolve({
  183. value: undefined,
  184. done: true
  185. });
  186. }
  187. var requestCapability = createPromiseCapability();
  188. this._requests.push(requestCapability);
  189. return requestCapability.promise;
  190. },
  191. cancel: function PDFWorkerStreamReader_cancel(reason) {
  192. this._done = true;
  193. this._requests.forEach(function (requestCapability) {
  194. requestCapability.resolve({
  195. value: undefined,
  196. done: true
  197. });
  198. });
  199. this._requests = [];
  200. }
  201. };
  202. function PDFWorkerStreamRangeReader(stream, begin, end) {
  203. this._stream = stream;
  204. this._begin = begin;
  205. this._end = end;
  206. this._queuedChunk = null;
  207. this._requests = [];
  208. this._done = false;
  209. this.onProgress = null;
  210. }
  211. PDFWorkerStreamRangeReader.prototype = {
  212. _enqueue: function PDFWorkerStreamRangeReader_enqueue(chunk) {
  213. if (this._done) {
  214. return;
  215. }
  216. if (this._requests.length === 0) {
  217. this._queuedChunk = chunk;
  218. } else {
  219. var requestsCapability = this._requests.shift();
  220. requestsCapability.resolve({
  221. value: chunk,
  222. done: false
  223. });
  224. this._requests.forEach(function (requestCapability) {
  225. requestCapability.resolve({
  226. value: undefined,
  227. done: true
  228. });
  229. });
  230. this._requests = [];
  231. }
  232. this._done = true;
  233. this._stream._removeRangeReader(this);
  234. },
  235. get isStreamingSupported() {
  236. return false;
  237. },
  238. read: function PDFWorkerStreamRangeReader_read() {
  239. if (this._queuedChunk) {
  240. return Promise.resolve({
  241. value: this._queuedChunk,
  242. done: false
  243. });
  244. }
  245. if (this._done) {
  246. return Promise.resolve({
  247. value: undefined,
  248. done: true
  249. });
  250. }
  251. var requestCapability = createPromiseCapability();
  252. this._requests.push(requestCapability);
  253. return requestCapability.promise;
  254. },
  255. cancel: function PDFWorkerStreamRangeReader_cancel(reason) {
  256. this._done = true;
  257. this._requests.forEach(function (requestCapability) {
  258. requestCapability.resolve({
  259. value: undefined,
  260. done: true
  261. });
  262. });
  263. this._requests = [];
  264. this._stream._removeRangeReader(this);
  265. }
  266. };
  267. return PDFWorkerStream;
  268. }();
  269. var PDFNetworkStream;
  270. function setPDFNetworkStreamClass(cls) {
  271. PDFNetworkStream = cls;
  272. }
  273. var WorkerMessageHandler = {
  274. setup: function setup(handler, port) {
  275. var testMessageProcessed = false;
  276. handler.on('test', function wphSetupTest(data) {
  277. if (testMessageProcessed) {
  278. return;
  279. }
  280. testMessageProcessed = true;
  281. if (!(data instanceof Uint8Array)) {
  282. handler.send('test', 'main', false);
  283. return;
  284. }
  285. var supportTransfers = data[0] === 255;
  286. handler.postMessageTransfers = supportTransfers;
  287. var xhr = new XMLHttpRequest();
  288. var responseExists = 'response' in xhr;
  289. try {
  290. xhr.responseType;
  291. } catch (e) {
  292. responseExists = false;
  293. }
  294. if (!responseExists) {
  295. handler.send('test', false);
  296. return;
  297. }
  298. handler.send('test', {
  299. supportTypedArray: true,
  300. supportTransfers: supportTransfers
  301. });
  302. });
  303. handler.on('configure', function wphConfigure(data) {
  304. setVerbosityLevel(data.verbosity);
  305. });
  306. handler.on('GetDocRequest', function wphSetupDoc(data) {
  307. return WorkerMessageHandler.createDocumentHandler(data, port);
  308. });
  309. },
  310. createDocumentHandler: function createDocumentHandler(docParams, port) {
  311. var pdfManager;
  312. var terminated = false;
  313. var cancelXHRs = null;
  314. var WorkerTasks = [];
  315. var docId = docParams.docId;
  316. var docBaseUrl = docParams.docBaseUrl;
  317. var workerHandlerName = docParams.docId + '_worker';
  318. var handler = new MessageHandler(workerHandlerName, docId, port);
  319. handler.postMessageTransfers = docParams.postMessageTransfers;
  320. function ensureNotTerminated() {
  321. if (terminated) {
  322. throw new Error('Worker was terminated');
  323. }
  324. }
  325. function startWorkerTask(task) {
  326. WorkerTasks.push(task);
  327. }
  328. function finishWorkerTask(task) {
  329. task.finish();
  330. var i = WorkerTasks.indexOf(task);
  331. WorkerTasks.splice(i, 1);
  332. }
  333. function loadDocument(recoveryMode) {
  334. var loadDocumentCapability = createPromiseCapability();
  335. var parseSuccess = function parseSuccess() {
  336. var numPagesPromise = pdfManager.ensureDoc('numPages');
  337. var fingerprintPromise = pdfManager.ensureDoc('fingerprint');
  338. var encryptedPromise = pdfManager.ensureXRef('encrypt');
  339. Promise.all([numPagesPromise, fingerprintPromise, encryptedPromise]).then(function onDocReady(results) {
  340. var doc = {
  341. numPages: results[0],
  342. fingerprint: results[1],
  343. encrypted: !!results[2]
  344. };
  345. loadDocumentCapability.resolve(doc);
  346. }, parseFailure);
  347. };
  348. var parseFailure = function parseFailure(e) {
  349. loadDocumentCapability.reject(e);
  350. };
  351. pdfManager.ensureDoc('checkHeader', []).then(function () {
  352. pdfManager.ensureDoc('parseStartXRef', []).then(function () {
  353. pdfManager.ensureDoc('parse', [recoveryMode]).then(parseSuccess, parseFailure);
  354. }, parseFailure);
  355. }, parseFailure);
  356. return loadDocumentCapability.promise;
  357. }
  358. function getPdfManager(data, evaluatorOptions) {
  359. var pdfManagerCapability = createPromiseCapability();
  360. var pdfManager;
  361. var source = data.source;
  362. if (source.data) {
  363. try {
  364. pdfManager = new LocalPdfManager(docId, source.data, source.password, evaluatorOptions, docBaseUrl);
  365. pdfManagerCapability.resolve(pdfManager);
  366. } catch (ex) {
  367. pdfManagerCapability.reject(ex);
  368. }
  369. return pdfManagerCapability.promise;
  370. }
  371. var pdfStream;
  372. try {
  373. if (source.chunkedViewerLoading) {
  374. pdfStream = new PDFWorkerStream(source, handler);
  375. } else {
  376. assert(PDFNetworkStream, 'pdfjs/core/network module is not loaded');
  377. pdfStream = new PDFNetworkStream(data);
  378. }
  379. } catch (ex) {
  380. pdfManagerCapability.reject(ex);
  381. return pdfManagerCapability.promise;
  382. }
  383. var fullRequest = pdfStream.getFullReader();
  384. fullRequest.headersReady.then(function () {
  385. if (!fullRequest.isStreamingSupported || !fullRequest.isRangeSupported) {
  386. fullRequest.onProgress = function (evt) {
  387. handler.send('DocProgress', {
  388. loaded: evt.loaded,
  389. total: evt.total
  390. });
  391. };
  392. }
  393. if (!fullRequest.isRangeSupported) {
  394. return;
  395. }
  396. var disableAutoFetch = source.disableAutoFetch || fullRequest.isStreamingSupported;
  397. pdfManager = new NetworkPdfManager(docId, pdfStream, {
  398. msgHandler: handler,
  399. url: source.url,
  400. password: source.password,
  401. length: fullRequest.contentLength,
  402. disableAutoFetch: disableAutoFetch,
  403. rangeChunkSize: source.rangeChunkSize
  404. }, evaluatorOptions, docBaseUrl);
  405. pdfManagerCapability.resolve(pdfManager);
  406. cancelXHRs = null;
  407. }).catch(function (reason) {
  408. pdfManagerCapability.reject(reason);
  409. cancelXHRs = null;
  410. });
  411. var cachedChunks = [],
  412. loaded = 0;
  413. var flushChunks = function flushChunks() {
  414. var pdfFile = arraysToBytes(cachedChunks);
  415. if (source.length && pdfFile.length !== source.length) {
  416. warn('reported HTTP length is different from actual');
  417. }
  418. try {
  419. pdfManager = new LocalPdfManager(docId, pdfFile, source.password, evaluatorOptions, docBaseUrl);
  420. pdfManagerCapability.resolve(pdfManager);
  421. } catch (ex) {
  422. pdfManagerCapability.reject(ex);
  423. }
  424. cachedChunks = [];
  425. };
  426. var readPromise = new Promise(function (resolve, reject) {
  427. var readChunk = function readChunk(chunk) {
  428. try {
  429. ensureNotTerminated();
  430. if (chunk.done) {
  431. if (!pdfManager) {
  432. flushChunks();
  433. }
  434. cancelXHRs = null;
  435. return;
  436. }
  437. var data = chunk.value;
  438. loaded += arrayByteLength(data);
  439. if (!fullRequest.isStreamingSupported) {
  440. handler.send('DocProgress', {
  441. loaded: loaded,
  442. total: Math.max(loaded, fullRequest.contentLength || 0)
  443. });
  444. }
  445. if (pdfManager) {
  446. pdfManager.sendProgressiveData(data);
  447. } else {
  448. cachedChunks.push(data);
  449. }
  450. fullRequest.read().then(readChunk, reject);
  451. } catch (e) {
  452. reject(e);
  453. }
  454. };
  455. fullRequest.read().then(readChunk, reject);
  456. });
  457. readPromise.catch(function (e) {
  458. pdfManagerCapability.reject(e);
  459. cancelXHRs = null;
  460. });
  461. cancelXHRs = function cancelXHRs() {
  462. pdfStream.cancelAllRequests('abort');
  463. };
  464. return pdfManagerCapability.promise;
  465. }
  466. function setupDoc(data) {
  467. function onSuccess(doc) {
  468. ensureNotTerminated();
  469. handler.send('GetDoc', { pdfInfo: doc });
  470. }
  471. function onFailure(e) {
  472. if (e instanceof PasswordException) {
  473. var task = new WorkerTask('PasswordException: response ' + e.code);
  474. startWorkerTask(task);
  475. handler.sendWithPromise('PasswordRequest', e).then(function (data) {
  476. finishWorkerTask(task);
  477. pdfManager.updatePassword(data.password);
  478. pdfManagerReady();
  479. }).catch(function (ex) {
  480. finishWorkerTask(task);
  481. handler.send('PasswordException', ex);
  482. }.bind(null, e));
  483. } else if (e instanceof InvalidPDFException) {
  484. handler.send('InvalidPDF', e);
  485. } else if (e instanceof MissingPDFException) {
  486. handler.send('MissingPDF', e);
  487. } else if (e instanceof UnexpectedResponseException) {
  488. handler.send('UnexpectedResponse', e);
  489. } else {
  490. handler.send('UnknownError', new UnknownErrorException(e.message, e.toString()));
  491. }
  492. }
  493. function pdfManagerReady() {
  494. ensureNotTerminated();
  495. loadDocument(false).then(onSuccess, function loadFailure(ex) {
  496. ensureNotTerminated();
  497. if (!(ex instanceof XRefParseException)) {
  498. onFailure(ex);
  499. return;
  500. }
  501. pdfManager.requestLoadedStream();
  502. pdfManager.onLoadedStream().then(function () {
  503. ensureNotTerminated();
  504. loadDocument(true).then(onSuccess, onFailure);
  505. });
  506. }, onFailure);
  507. }
  508. ensureNotTerminated();
  509. var evaluatorOptions = {
  510. forceDataSchema: data.disableCreateObjectURL,
  511. maxImageSize: data.maxImageSize === undefined ? -1 : data.maxImageSize,
  512. disableFontFace: data.disableFontFace,
  513. disableNativeImageDecoder: data.disableNativeImageDecoder,
  514. ignoreErrors: data.ignoreErrors
  515. };
  516. getPdfManager(data, evaluatorOptions).then(function (newPdfManager) {
  517. if (terminated) {
  518. newPdfManager.terminate();
  519. throw new Error('Worker was terminated');
  520. }
  521. pdfManager = newPdfManager;
  522. handler.send('PDFManagerReady', null);
  523. pdfManager.onLoadedStream().then(function (stream) {
  524. handler.send('DataLoaded', { length: stream.bytes.byteLength });
  525. });
  526. }).then(pdfManagerReady, onFailure);
  527. }
  528. handler.on('GetPage', function wphSetupGetPage(data) {
  529. return pdfManager.getPage(data.pageIndex).then(function (page) {
  530. var rotatePromise = pdfManager.ensure(page, 'rotate');
  531. var refPromise = pdfManager.ensure(page, 'ref');
  532. var userUnitPromise = pdfManager.ensure(page, 'userUnit');
  533. var viewPromise = pdfManager.ensure(page, 'view');
  534. return Promise.all([rotatePromise, refPromise, userUnitPromise, viewPromise]).then(function (results) {
  535. return {
  536. rotate: results[0],
  537. ref: results[1],
  538. userUnit: results[2],
  539. view: results[3]
  540. };
  541. });
  542. });
  543. });
  544. handler.on('GetPageIndex', function wphSetupGetPageIndex(data) {
  545. var ref = new Ref(data.ref.num, data.ref.gen);
  546. var catalog = pdfManager.pdfDocument.catalog;
  547. return catalog.getPageIndex(ref);
  548. });
  549. handler.on('GetDestinations', function wphSetupGetDestinations(data) {
  550. return pdfManager.ensureCatalog('destinations');
  551. });
  552. handler.on('GetDestination', function wphSetupGetDestination(data) {
  553. return pdfManager.ensureCatalog('getDestination', [data.id]);
  554. });
  555. handler.on('GetPageLabels', function wphSetupGetPageLabels(data) {
  556. return pdfManager.ensureCatalog('pageLabels');
  557. });
  558. handler.on('GetAttachments', function wphSetupGetAttachments(data) {
  559. return pdfManager.ensureCatalog('attachments');
  560. });
  561. handler.on('GetJavaScript', function wphSetupGetJavaScript(data) {
  562. return pdfManager.ensureCatalog('javaScript');
  563. });
  564. handler.on('GetOutline', function wphSetupGetOutline(data) {
  565. return pdfManager.ensureCatalog('documentOutline');
  566. });
  567. handler.on('GetMetadata', function wphSetupGetMetadata(data) {
  568. return Promise.all([pdfManager.ensureDoc('documentInfo'), pdfManager.ensureCatalog('metadata')]);
  569. });
  570. handler.on('GetData', function wphSetupGetData(data) {
  571. pdfManager.requestLoadedStream();
  572. return pdfManager.onLoadedStream().then(function (stream) {
  573. return stream.bytes;
  574. });
  575. });
  576. handler.on('GetStats', function wphSetupGetStats(data) {
  577. return pdfManager.pdfDocument.xref.stats;
  578. });
  579. handler.on('GetAnnotations', function wphSetupGetAnnotations(data) {
  580. return pdfManager.getPage(data.pageIndex).then(function (page) {
  581. return pdfManager.ensure(page, 'getAnnotationsData', [data.intent]);
  582. });
  583. });
  584. handler.on('RenderPageRequest', function wphSetupRenderPage(data) {
  585. var pageIndex = data.pageIndex;
  586. pdfManager.getPage(pageIndex).then(function (page) {
  587. var task = new WorkerTask('RenderPageRequest: page ' + pageIndex);
  588. startWorkerTask(task);
  589. var pageNum = pageIndex + 1;
  590. var start = Date.now();
  591. page.getOperatorList({
  592. handler: handler,
  593. task: task,
  594. intent: data.intent,
  595. renderInteractiveForms: data.renderInteractiveForms
  596. }).then(function (operatorList) {
  597. finishWorkerTask(task);
  598. info('page=' + pageNum + ' - getOperatorList: time=' + (Date.now() - start) + 'ms, len=' + operatorList.totalLength);
  599. }, function (e) {
  600. finishWorkerTask(task);
  601. if (task.terminated) {
  602. return;
  603. }
  604. handler.send('UnsupportedFeature', { featureId: UNSUPPORTED_FEATURES.unknown });
  605. var minimumStackMessage = 'worker.js: while trying to getPage() and getOperatorList()';
  606. var wrappedException;
  607. if (typeof e === 'string') {
  608. wrappedException = {
  609. message: e,
  610. stack: minimumStackMessage
  611. };
  612. } else if ((typeof e === 'undefined' ? 'undefined' : _typeof(e)) === 'object') {
  613. wrappedException = {
  614. message: e.message || e.toString(),
  615. stack: e.stack || minimumStackMessage
  616. };
  617. } else {
  618. wrappedException = {
  619. message: 'Unknown exception type: ' + (typeof e === 'undefined' ? 'undefined' : _typeof(e)),
  620. stack: minimumStackMessage
  621. };
  622. }
  623. handler.send('PageError', {
  624. pageNum: pageNum,
  625. error: wrappedException,
  626. intent: data.intent
  627. });
  628. });
  629. });
  630. }, this);
  631. handler.on('GetTextContent', function wphExtractText(data) {
  632. var pageIndex = data.pageIndex;
  633. return pdfManager.getPage(pageIndex).then(function (page) {
  634. var task = new WorkerTask('GetTextContent: page ' + pageIndex);
  635. startWorkerTask(task);
  636. var pageNum = pageIndex + 1;
  637. var start = Date.now();
  638. return page.extractTextContent({
  639. handler: handler,
  640. task: task,
  641. normalizeWhitespace: data.normalizeWhitespace,
  642. combineTextItems: data.combineTextItems
  643. }).then(function (textContent) {
  644. finishWorkerTask(task);
  645. info('text indexing: page=' + pageNum + ' - time=' + (Date.now() - start) + 'ms');
  646. return textContent;
  647. }, function (reason) {
  648. finishWorkerTask(task);
  649. if (task.terminated) {
  650. return;
  651. }
  652. throw reason;
  653. });
  654. });
  655. });
  656. handler.on('Cleanup', function wphCleanup(data) {
  657. return pdfManager.cleanup();
  658. });
  659. handler.on('Terminate', function wphTerminate(data) {
  660. terminated = true;
  661. if (pdfManager) {
  662. pdfManager.terminate();
  663. pdfManager = null;
  664. }
  665. if (cancelXHRs) {
  666. cancelXHRs();
  667. }
  668. var waitOn = [];
  669. WorkerTasks.forEach(function (task) {
  670. waitOn.push(task.finished);
  671. task.terminate();
  672. });
  673. return Promise.all(waitOn).then(function () {
  674. handler.destroy();
  675. handler = null;
  676. });
  677. });
  678. handler.on('Ready', function wphReady(data) {
  679. setupDoc(docParams);
  680. docParams = null;
  681. });
  682. return workerHandlerName;
  683. },
  684. initializeFromPort: function initializeFromPort(port) {
  685. var handler = new MessageHandler('worker', 'main', port);
  686. WorkerMessageHandler.setup(handler, port);
  687. handler.send('ready', null);
  688. }
  689. };
  690. function isMessagePort(maybePort) {
  691. return typeof maybePort.postMessage === 'function' && 'onmessage' in maybePort;
  692. }
  693. if (typeof window === 'undefined' && !isNodeJS() && typeof self !== 'undefined' && isMessagePort(self)) {
  694. WorkerMessageHandler.initializeFromPort(self);
  695. }
  696. exports.setPDFNetworkStreamClass = setPDFNetworkStreamClass;
  697. exports.WorkerTask = WorkerTask;
  698. exports.WorkerMessageHandler = WorkerMessageHandler;