Adds Streams API support for networking task of PDF.js project.
network.js file moved to main thread and `PDFNetworkStream` implemented at worker thread, that is used to ask for data whenever worker needs.
This commit is contained in:
parent
bd8c12119a
commit
109106794d
@ -198,218 +198,120 @@ IPDFStreamRangeReader.prototype = {
|
|||||||
|
|
||||||
/** @implements {IPDFStream} */
|
/** @implements {IPDFStream} */
|
||||||
var PDFWorkerStream = (function PDFWorkerStreamClosure() {
|
var PDFWorkerStream = (function PDFWorkerStreamClosure() {
|
||||||
function PDFWorkerStream(params, msgHandler) {
|
function PDFWorkerStream(msgHandler) {
|
||||||
this._queuedChunks = [];
|
|
||||||
var initialData = params.initialData;
|
|
||||||
if (initialData && initialData.length > 0) {
|
|
||||||
this._queuedChunks.push(initialData);
|
|
||||||
}
|
|
||||||
this._msgHandler = msgHandler;
|
this._msgHandler = msgHandler;
|
||||||
|
this._contentLength = null;
|
||||||
this._isRangeSupported = !(params.disableRange);
|
|
||||||
this._isStreamingSupported = !(params.disableStream);
|
|
||||||
this._contentLength = params.length;
|
|
||||||
|
|
||||||
this._fullRequestReader = null;
|
this._fullRequestReader = null;
|
||||||
this._rangeReaders = [];
|
this._rangeRequestReaders = [];
|
||||||
|
|
||||||
msgHandler.on('OnDataRange', this._onReceiveData.bind(this));
|
|
||||||
msgHandler.on('OnDataProgress', this._onProgress.bind(this));
|
|
||||||
}
|
}
|
||||||
PDFWorkerStream.prototype = {
|
PDFWorkerStream.prototype = {
|
||||||
_onReceiveData: function PDFWorkerStream_onReceiveData(args) {
|
getFullReader() {
|
||||||
if (args.begin === undefined) {
|
|
||||||
if (this._fullRequestReader) {
|
|
||||||
this._fullRequestReader._enqueue(args.chunk);
|
|
||||||
} else {
|
|
||||||
this._queuedChunks.push(args.chunk);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
var found = this._rangeReaders.some(function (rangeReader) {
|
|
||||||
if (rangeReader._begin !== args.begin) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
rangeReader._enqueue(args.chunk);
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
assert(found);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_onProgress: function PDFWorkerStream_onProgress(evt) {
|
|
||||||
if (this._rangeReaders.length > 0) {
|
|
||||||
// Reporting to first range reader.
|
|
||||||
var firstReader = this._rangeReaders[0];
|
|
||||||
if (firstReader.onProgress) {
|
|
||||||
firstReader.onProgress({ loaded: evt.loaded, });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
_removeRangeReader: function PDFWorkerStream_removeRangeReader(reader) {
|
|
||||||
var i = this._rangeReaders.indexOf(reader);
|
|
||||||
if (i >= 0) {
|
|
||||||
this._rangeReaders.splice(i, 1);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
getFullReader: function PDFWorkerStream_getFullReader() {
|
|
||||||
assert(!this._fullRequestReader);
|
assert(!this._fullRequestReader);
|
||||||
var queuedChunks = this._queuedChunks;
|
this._fullRequestReader = new PDFWorkerStreamReader(this._msgHandler);
|
||||||
this._queuedChunks = null;
|
return this._fullRequestReader;
|
||||||
return new PDFWorkerStreamReader(this, queuedChunks);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
getRangeReader: function PDFWorkerStream_getRangeReader(begin, end) {
|
getRangeReader(begin, end) {
|
||||||
var reader = new PDFWorkerStreamRangeReader(this, begin, end);
|
let reader = new PDFWorkerStreamRangeReader(begin, end, this._msgHandler);
|
||||||
this._msgHandler.send('RequestDataRange', { begin, end, });
|
this._rangeRequestReaders.push(reader);
|
||||||
this._rangeReaders.push(reader);
|
|
||||||
return reader;
|
return reader;
|
||||||
},
|
},
|
||||||
|
|
||||||
cancelAllRequests: function PDFWorkerStream_cancelAllRequests(reason) {
|
cancelAllRequests(reason) {
|
||||||
if (this._fullRequestReader) {
|
if (this._fullRequestReader) {
|
||||||
this._fullRequestReader.cancel(reason);
|
this._fullRequestReader.cancel(reason);
|
||||||
}
|
}
|
||||||
var readers = this._rangeReaders.slice(0);
|
let readers = this._rangeRequestReaders.slice(0);
|
||||||
readers.forEach(function (rangeReader) {
|
readers.forEach(function (reader) {
|
||||||
rangeReader.cancel(reason);
|
reader.cancel(reason);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @implements {IPDFStreamReader} */
|
/** @implements {IPDFStreamReader} */
|
||||||
function PDFWorkerStreamReader(stream, queuedChunks) {
|
function PDFWorkerStreamReader(msgHandler) {
|
||||||
this._stream = stream;
|
this._msgHandler = msgHandler;
|
||||||
this._done = false;
|
|
||||||
this._queuedChunks = queuedChunks || [];
|
|
||||||
this._requests = [];
|
|
||||||
this._headersReady = Promise.resolve();
|
|
||||||
stream._fullRequestReader = this;
|
|
||||||
|
|
||||||
this.onProgress = null; // not used
|
this._contentLength = null;
|
||||||
|
this._isRangeSupported = false;
|
||||||
|
this._isStreamingSupported = false;
|
||||||
|
|
||||||
|
let readableStream = this._msgHandler.sendWithStream('GetReader');
|
||||||
|
|
||||||
|
this._reader = readableStream.getReader();
|
||||||
|
|
||||||
|
this._headersReady = this._msgHandler.sendWithPromise('ReaderHeadersReady').
|
||||||
|
then((data) => {
|
||||||
|
this._isStreamingSupported = data.isStreamingSupported;
|
||||||
|
this._isRangeSupported = data.isRangeSupported;
|
||||||
|
this._contentLength = data.contentLength;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
PDFWorkerStreamReader.prototype = {
|
PDFWorkerStreamReader.prototype = {
|
||||||
_enqueue: function PDFWorkerStreamReader_enqueue(chunk) {
|
|
||||||
if (this._done) {
|
|
||||||
return; // ignore new data
|
|
||||||
}
|
|
||||||
if (this._requests.length > 0) {
|
|
||||||
var requestCapability = this._requests.shift();
|
|
||||||
requestCapability.resolve({ value: chunk, done: false, });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this._queuedChunks.push(chunk);
|
|
||||||
},
|
|
||||||
|
|
||||||
get headersReady() {
|
get headersReady() {
|
||||||
return this._headersReady;
|
return this._headersReady;
|
||||||
},
|
},
|
||||||
|
|
||||||
get isRangeSupported() {
|
get contentLength() {
|
||||||
return this._stream._isRangeSupported;
|
return this._contentLength;
|
||||||
},
|
},
|
||||||
|
|
||||||
get isStreamingSupported() {
|
get isStreamingSupported() {
|
||||||
return this._stream._isStreamingSupported;
|
return this._isStreamingSupported;
|
||||||
},
|
},
|
||||||
|
|
||||||
get contentLength() {
|
get isRangeSupported() {
|
||||||
return this._stream._contentLength;
|
return this._isRangeSupported;
|
||||||
},
|
},
|
||||||
|
|
||||||
read: function PDFWorkerStreamReader_read() {
|
read() {
|
||||||
if (this._queuedChunks.length > 0) {
|
return this._reader.read().then(function({ value, done, }) {
|
||||||
var chunk = this._queuedChunks.shift();
|
if (done) {
|
||||||
return Promise.resolve({ value: chunk, done: false, });
|
return { value: undefined, done: true, };
|
||||||
}
|
}
|
||||||
if (this._done) {
|
// `value` is wrapped into Uint8Array, we need to
|
||||||
return Promise.resolve({ value: undefined, done: true, });
|
// unwrap it to ArrayBuffer for further processing.
|
||||||
}
|
return { value: value.buffer, done: false, };
|
||||||
var requestCapability = createPromiseCapability();
|
|
||||||
this._requests.push(requestCapability);
|
|
||||||
return requestCapability.promise;
|
|
||||||
},
|
|
||||||
|
|
||||||
cancel: function PDFWorkerStreamReader_cancel(reason) {
|
|
||||||
this._done = true;
|
|
||||||
this._requests.forEach(function (requestCapability) {
|
|
||||||
requestCapability.resolve({ value: undefined, done: true, });
|
|
||||||
});
|
});
|
||||||
this._requests = [];
|
},
|
||||||
|
|
||||||
|
cancel(reason) {
|
||||||
|
this._reader.cancel(reason);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/** @implements {IPDFStreamRangeReader} */
|
/** @implements {IPDFStreamRangeReader} */
|
||||||
function PDFWorkerStreamRangeReader(stream, begin, end) {
|
function PDFWorkerStreamRangeReader(begin, end, msgHandler) {
|
||||||
this._stream = stream;
|
this._msgHandler = msgHandler;
|
||||||
this._begin = begin;
|
|
||||||
this._end = end;
|
|
||||||
this._queuedChunk = null;
|
|
||||||
this._requests = [];
|
|
||||||
this._done = false;
|
|
||||||
|
|
||||||
this.onProgress = null;
|
this.onProgress = null;
|
||||||
|
|
||||||
|
let readableStream = this._msgHandler.sendWithStream('GetRangeReader',
|
||||||
|
{ begin, end, });
|
||||||
|
|
||||||
|
this._reader = readableStream.getReader();
|
||||||
}
|
}
|
||||||
PDFWorkerStreamRangeReader.prototype = {
|
PDFWorkerStreamRangeReader.prototype = {
|
||||||
_enqueue: function PDFWorkerStreamRangeReader_enqueue(chunk) {
|
|
||||||
if (this._done) {
|
|
||||||
return; // ignore new data
|
|
||||||
}
|
|
||||||
if (this._requests.length === 0) {
|
|
||||||
this._queuedChunk = chunk;
|
|
||||||
} else {
|
|
||||||
var requestsCapability = this._requests.shift();
|
|
||||||
requestsCapability.resolve({ value: chunk, done: false, });
|
|
||||||
this._requests.forEach(function (requestCapability) {
|
|
||||||
requestCapability.resolve({ value: undefined, done: true, });
|
|
||||||
});
|
|
||||||
this._requests = [];
|
|
||||||
}
|
|
||||||
this._done = true;
|
|
||||||
this._stream._removeRangeReader(this);
|
|
||||||
},
|
|
||||||
|
|
||||||
get isStreamingSupported() {
|
get isStreamingSupported() {
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
|
|
||||||
read: function PDFWorkerStreamRangeReader_read() {
|
read() {
|
||||||
if (this._queuedChunk) {
|
return this._reader.read().then(function({ value, done, }) {
|
||||||
return Promise.resolve({ value: this._queuedChunk, done: false, });
|
if (done) {
|
||||||
}
|
return { value: undefined, done: true, };
|
||||||
if (this._done) {
|
}
|
||||||
return Promise.resolve({ value: undefined, done: true, });
|
return { value: value.buffer, done: false, };
|
||||||
}
|
});
|
||||||
var requestCapability = createPromiseCapability();
|
|
||||||
this._requests.push(requestCapability);
|
|
||||||
return requestCapability.promise;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
cancel: function PDFWorkerStreamRangeReader_cancel(reason) {
|
cancel(reason) {
|
||||||
this._done = true;
|
this._reader.cancel(reason);
|
||||||
this._requests.forEach(function (requestCapability) {
|
|
||||||
requestCapability.resolve({ value: undefined, done: true, });
|
|
||||||
});
|
|
||||||
this._requests = [];
|
|
||||||
this._stream._removeRangeReader(this);
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
return PDFWorkerStream;
|
return PDFWorkerStream;
|
||||||
})();
|
})();
|
||||||
|
|
||||||
/** @type IPDFStream */
|
|
||||||
var PDFNetworkStream;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets PDFNetworkStream class to be used as alternative PDF data transport.
|
|
||||||
* @param {IPDFStream} cls - the PDF data transport.
|
|
||||||
*/
|
|
||||||
function setPDFNetworkStreamClass(cls) {
|
|
||||||
PDFNetworkStream = cls;
|
|
||||||
}
|
|
||||||
|
|
||||||
var WorkerMessageHandler = {
|
var WorkerMessageHandler = {
|
||||||
setup(handler, port) {
|
setup(handler, port) {
|
||||||
var testMessageProcessed = false;
|
var testMessageProcessed = false;
|
||||||
@ -536,16 +438,9 @@ var WorkerMessageHandler = {
|
|||||||
return pdfManagerCapability.promise;
|
return pdfManagerCapability.promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
var pdfStream;
|
var pdfStream, cachedChunks = [];
|
||||||
try {
|
try {
|
||||||
if (source.chunkedViewerLoading) {
|
pdfStream = new PDFWorkerStream(handler);
|
||||||
pdfStream = new PDFWorkerStream(source, handler);
|
|
||||||
} else {
|
|
||||||
if (!PDFNetworkStream) {
|
|
||||||
throw new Error('./network module is not loaded');
|
|
||||||
}
|
|
||||||
pdfStream = new PDFNetworkStream(data);
|
|
||||||
}
|
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
pdfManagerCapability.reject(ex);
|
pdfManagerCapability.reject(ex);
|
||||||
return pdfManagerCapability.promise;
|
return pdfManagerCapability.promise;
|
||||||
@ -553,18 +448,6 @@ var WorkerMessageHandler = {
|
|||||||
|
|
||||||
var fullRequest = pdfStream.getFullReader();
|
var fullRequest = pdfStream.getFullReader();
|
||||||
fullRequest.headersReady.then(function () {
|
fullRequest.headersReady.then(function () {
|
||||||
if (!fullRequest.isStreamingSupported ||
|
|
||||||
!fullRequest.isRangeSupported) {
|
|
||||||
// If stream or range are disabled, it's our only way to report
|
|
||||||
// loading progress.
|
|
||||||
fullRequest.onProgress = function (evt) {
|
|
||||||
handler.send('DocProgress', {
|
|
||||||
loaded: evt.loaded,
|
|
||||||
total: evt.total,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!fullRequest.isRangeSupported) {
|
if (!fullRequest.isRangeSupported) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -580,6 +463,15 @@ var WorkerMessageHandler = {
|
|||||||
disableAutoFetch,
|
disableAutoFetch,
|
||||||
rangeChunkSize: source.rangeChunkSize,
|
rangeChunkSize: source.rangeChunkSize,
|
||||||
}, evaluatorOptions, docBaseUrl);
|
}, evaluatorOptions, docBaseUrl);
|
||||||
|
// There may be a chance that `pdfManager` is not initialized
|
||||||
|
// for first few runs of `readchunk` block of code. Be sure
|
||||||
|
// to send all cached chunks, if any, to chunked_stream via
|
||||||
|
// pdf_manager.
|
||||||
|
for (let i = 0; i < cachedChunks.length; i++) {
|
||||||
|
pdfManager.sendProgressiveData(cachedChunks[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
cachedChunks = [];
|
||||||
pdfManagerCapability.resolve(pdfManager);
|
pdfManagerCapability.resolve(pdfManager);
|
||||||
cancelXHRs = null;
|
cancelXHRs = null;
|
||||||
}).catch(function (reason) {
|
}).catch(function (reason) {
|
||||||
@ -587,7 +479,7 @@ var WorkerMessageHandler = {
|
|||||||
cancelXHRs = null;
|
cancelXHRs = null;
|
||||||
});
|
});
|
||||||
|
|
||||||
var cachedChunks = [], loaded = 0;
|
var loaded = 0;
|
||||||
var flushChunks = function () {
|
var flushChunks = function () {
|
||||||
var pdfFile = arraysToBytes(cachedChunks);
|
var pdfFile = arraysToBytes(cachedChunks);
|
||||||
if (source.length && pdfFile.length !== source.length) {
|
if (source.length && pdfFile.length !== source.length) {
|
||||||
@ -969,7 +861,6 @@ if (typeof window === 'undefined' && !isNodeJS() &&
|
|||||||
}
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
setPDFNetworkStreamClass,
|
|
||||||
WorkerTask,
|
WorkerTask,
|
||||||
WorkerMessageHandler,
|
WorkerMessageHandler,
|
||||||
};
|
};
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
/* globals requirejs, __non_webpack_require__ */
|
/* globals requirejs, __non_webpack_require__ */
|
||||||
|
|
||||||
import {
|
import {
|
||||||
createPromiseCapability, deprecated, getVerbosityLevel, globalScope,
|
assert, createPromiseCapability, deprecated, getVerbosityLevel, globalScope,
|
||||||
info, InvalidPDFException, isArray, isArrayBuffer, isInt, isSameOrigin,
|
info, InvalidPDFException, isArray, isArrayBuffer, isInt, isSameOrigin,
|
||||||
loadJpegStream, MessageHandler, MissingPDFException, NativeImageDecoding,
|
loadJpegStream, MessageHandler, MissingPDFException, NativeImageDecoding,
|
||||||
PageViewport, PasswordException, StatTimer, stringToBytes,
|
PageViewport, PasswordException, StatTimer, stringToBytes,
|
||||||
@ -28,6 +28,7 @@ import {
|
|||||||
import { FontFaceObject, FontLoader } from './font_loader';
|
import { FontFaceObject, FontLoader } from './font_loader';
|
||||||
import { CanvasGraphics } from './canvas';
|
import { CanvasGraphics } from './canvas';
|
||||||
import { Metadata } from './metadata';
|
import { Metadata } from './metadata';
|
||||||
|
import { PDFDataTransportStream } from './transport_stream';
|
||||||
|
|
||||||
var DEFAULT_RANGE_CHUNK_SIZE = 65536; // 2^16 = 65536
|
var DEFAULT_RANGE_CHUNK_SIZE = 65536; // 2^16 = 65536
|
||||||
|
|
||||||
@ -80,6 +81,17 @@ if (typeof PDFJSDev !== 'undefined' &&
|
|||||||
}) : null;
|
}) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @type IPDFStream */
|
||||||
|
var PDFNetworkStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets PDFNetworkStream class to be used as alternative PDF data transport.
|
||||||
|
* @param {IPDFStream} cls - the PDF data transport.
|
||||||
|
*/
|
||||||
|
function setPDFNetworkStreamClass(cls) {
|
||||||
|
PDFNetworkStream = cls;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Document initialization / loading parameters object.
|
* Document initialization / loading parameters object.
|
||||||
*
|
*
|
||||||
@ -281,8 +293,20 @@ function getDocument(src, pdfDataRangeTransport,
|
|||||||
if (task.destroyed) {
|
if (task.destroyed) {
|
||||||
throw new Error('Loading aborted');
|
throw new Error('Loading aborted');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let networkStream;
|
||||||
|
if (rangeTransport) {
|
||||||
|
networkStream = new PDFDataTransportStream(params, rangeTransport);
|
||||||
|
} else if (!params.data) {
|
||||||
|
networkStream = new PDFNetworkStream({
|
||||||
|
source: params,
|
||||||
|
disableRange: getDefaultSetting('disableRange'),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
var messageHandler = new MessageHandler(docId, workerId, worker.port);
|
var messageHandler = new MessageHandler(docId, workerId, worker.port);
|
||||||
var transport = new WorkerTransport(messageHandler, task, rangeTransport,
|
messageHandler.postMessageTransfers = worker.postMessageTransfers;
|
||||||
|
var transport = new WorkerTransport(messageHandler, task, networkStream,
|
||||||
CMapReaderFactory);
|
CMapReaderFactory);
|
||||||
task._transport = transport;
|
task._transport = transport;
|
||||||
messageHandler.send('Ready', null);
|
messageHandler.send('Ready', null);
|
||||||
@ -317,7 +341,6 @@ function _fetchDocument(worker, source, pdfDataRangeTransport, docId) {
|
|||||||
return worker.messageHandler.sendWithPromise('GetDocRequest', {
|
return worker.messageHandler.sendWithPromise('GetDocRequest', {
|
||||||
docId,
|
docId,
|
||||||
source,
|
source,
|
||||||
disableRange: getDefaultSetting('disableRange'),
|
|
||||||
maxImageSize: getDefaultSetting('maxImageSize'),
|
maxImageSize: getDefaultSetting('maxImageSize'),
|
||||||
disableFontFace: getDefaultSetting('disableFontFace'),
|
disableFontFace: getDefaultSetting('disableFontFace'),
|
||||||
disableCreateObjectURL: getDefaultSetting('disableCreateObjectURL'),
|
disableCreateObjectURL: getDefaultSetting('disableCreateObjectURL'),
|
||||||
@ -1238,9 +1261,7 @@ var PDFWorker = (function PDFWorkerClosure() {
|
|||||||
// pdf.worker.js file is needed.
|
// pdf.worker.js file is needed.
|
||||||
if (typeof PDFJSDev === 'undefined' || !PDFJSDev.test('PRODUCTION')) {
|
if (typeof PDFJSDev === 'undefined' || !PDFJSDev.test('PRODUCTION')) {
|
||||||
if (typeof SystemJS === 'object') {
|
if (typeof SystemJS === 'object') {
|
||||||
Promise.all([SystemJS.import('pdfjs/core/network'),
|
SystemJS.import('pdfjs/core/worker').then((worker) => {
|
||||||
SystemJS.import('pdfjs/core/worker')]).then((modules) => {
|
|
||||||
var worker = modules[1];
|
|
||||||
WorkerMessageHandler = worker.WorkerMessageHandler;
|
WorkerMessageHandler = worker.WorkerMessageHandler;
|
||||||
fakeWorkerFilesLoadedCapability.resolve(WorkerMessageHandler);
|
fakeWorkerFilesLoadedCapability.resolve(WorkerMessageHandler);
|
||||||
});
|
});
|
||||||
@ -1254,7 +1275,6 @@ var PDFWorker = (function PDFWorkerClosure() {
|
|||||||
}
|
}
|
||||||
} else if (PDFJSDev.test('SINGLE_FILE')) {
|
} else if (PDFJSDev.test('SINGLE_FILE')) {
|
||||||
var pdfjsCoreWorker = require('../core/worker.js');
|
var pdfjsCoreWorker = require('../core/worker.js');
|
||||||
require('../core/network.js');
|
|
||||||
WorkerMessageHandler = pdfjsCoreWorker.WorkerMessageHandler;
|
WorkerMessageHandler = pdfjsCoreWorker.WorkerMessageHandler;
|
||||||
fakeWorkerFilesLoadedCapability.resolve(WorkerMessageHandler);
|
fakeWorkerFilesLoadedCapability.resolve(WorkerMessageHandler);
|
||||||
} else {
|
} else {
|
||||||
@ -1285,6 +1305,7 @@ var PDFWorker = (function PDFWorkerClosure() {
|
|||||||
|
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.destroyed = false;
|
this.destroyed = false;
|
||||||
|
this.postMessageTransfers = true;
|
||||||
|
|
||||||
this._readyCapability = createPromiseCapability();
|
this._readyCapability = createPromiseCapability();
|
||||||
this._port = null;
|
this._port = null;
|
||||||
@ -1381,6 +1402,7 @@ var PDFWorker = (function PDFWorkerClosure() {
|
|||||||
this._port = worker;
|
this._port = worker;
|
||||||
this._webWorker = worker;
|
this._webWorker = worker;
|
||||||
if (!data.supportTransfers) {
|
if (!data.supportTransfers) {
|
||||||
|
this.postMessageTransfers = false;
|
||||||
isPostMessageTransfersDisabled = true;
|
isPostMessageTransfersDisabled = true;
|
||||||
}
|
}
|
||||||
this._readyCapability.resolve();
|
this._readyCapability.resolve();
|
||||||
@ -1513,11 +1535,10 @@ var PDFWorker = (function PDFWorkerClosure() {
|
|||||||
* @ignore
|
* @ignore
|
||||||
*/
|
*/
|
||||||
var WorkerTransport = (function WorkerTransportClosure() {
|
var WorkerTransport = (function WorkerTransportClosure() {
|
||||||
function WorkerTransport(messageHandler, loadingTask, pdfDataRangeTransport,
|
function WorkerTransport(messageHandler, loadingTask, networkStream,
|
||||||
CMapReaderFactory) {
|
CMapReaderFactory) {
|
||||||
this.messageHandler = messageHandler;
|
this.messageHandler = messageHandler;
|
||||||
this.loadingTask = loadingTask;
|
this.loadingTask = loadingTask;
|
||||||
this.pdfDataRangeTransport = pdfDataRangeTransport;
|
|
||||||
this.commonObjs = new PDFObjects();
|
this.commonObjs = new PDFObjects();
|
||||||
this.fontLoader = new FontLoader(loadingTask.docId);
|
this.fontLoader = new FontLoader(loadingTask.docId);
|
||||||
this.CMapReaderFactory = new CMapReaderFactory({
|
this.CMapReaderFactory = new CMapReaderFactory({
|
||||||
@ -1529,6 +1550,10 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
this.destroyCapability = null;
|
this.destroyCapability = null;
|
||||||
this._passwordCapability = null;
|
this._passwordCapability = null;
|
||||||
|
|
||||||
|
this._networkStream = networkStream;
|
||||||
|
this._fullReader = null;
|
||||||
|
this._lastProgress = null;
|
||||||
|
|
||||||
this.pageCache = [];
|
this.pageCache = [];
|
||||||
this.pagePromises = [];
|
this.pagePromises = [];
|
||||||
this.downloadInfoCapability = createPromiseCapability();
|
this.downloadInfoCapability = createPromiseCapability();
|
||||||
@ -1564,10 +1589,10 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
waitOn.push(terminated);
|
waitOn.push(terminated);
|
||||||
Promise.all(waitOn).then(() => {
|
Promise.all(waitOn).then(() => {
|
||||||
this.fontLoader.clear();
|
this.fontLoader.clear();
|
||||||
if (this.pdfDataRangeTransport) {
|
if (this._networkStream) {
|
||||||
this.pdfDataRangeTransport.abort();
|
this._networkStream.cancelAllRequests();
|
||||||
this.pdfDataRangeTransport = null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.messageHandler) {
|
if (this.messageHandler) {
|
||||||
this.messageHandler.destroy();
|
this.messageHandler.destroy();
|
||||||
this.messageHandler = null;
|
this.messageHandler = null;
|
||||||
@ -1581,32 +1606,92 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
var messageHandler = this.messageHandler;
|
var messageHandler = this.messageHandler;
|
||||||
var loadingTask = this.loadingTask;
|
var loadingTask = this.loadingTask;
|
||||||
|
|
||||||
var pdfDataRangeTransport = this.pdfDataRangeTransport;
|
messageHandler.on('GetReader', function(data, sink) {
|
||||||
if (pdfDataRangeTransport) {
|
assert(this._networkStream);
|
||||||
pdfDataRangeTransport.addRangeListener(function(begin, chunk) {
|
this._fullReader = this._networkStream.getFullReader();
|
||||||
messageHandler.send('OnDataRange', {
|
this._fullReader.onProgress = (evt) => {
|
||||||
begin,
|
this._lastProgress = {
|
||||||
chunk,
|
loaded: evt.loaded,
|
||||||
|
total: evt.total,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
sink.onPull = () => {
|
||||||
|
this._fullReader.read().then(function({ value, done, }) {
|
||||||
|
if (done) {
|
||||||
|
sink.close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
assert(isArrayBuffer(value));
|
||||||
|
// Enqueue data chunk into sink, and transfer it
|
||||||
|
// to other side as `Transferable` object.
|
||||||
|
sink.enqueue(new Uint8Array(value), 1, [value]);
|
||||||
|
}).catch((reason) => {
|
||||||
|
sink.error(reason);
|
||||||
});
|
});
|
||||||
});
|
};
|
||||||
|
|
||||||
pdfDataRangeTransport.addProgressListener(function(loaded) {
|
sink.onCancel = (reason) => {
|
||||||
messageHandler.send('OnDataProgress', {
|
this._fullReader.cancel(reason);
|
||||||
loaded,
|
};
|
||||||
|
}, this);
|
||||||
|
|
||||||
|
messageHandler.on('ReaderHeadersReady', function(data) {
|
||||||
|
let headersCapability = createPromiseCapability();
|
||||||
|
let fullReader = this._fullReader;
|
||||||
|
fullReader.headersReady.then(() => {
|
||||||
|
// If stream or range are disabled, it's our only way to report
|
||||||
|
// loading progress.
|
||||||
|
if (!fullReader.isStreamingSupported ||
|
||||||
|
!fullReader.isRangeSupported) {
|
||||||
|
if (this._lastProgress) {
|
||||||
|
let loadingTask = this.loadingTask;
|
||||||
|
if (loadingTask.onProgress) {
|
||||||
|
loadingTask.onProgress(this._lastProgress);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fullReader.onProgress = (evt) => {
|
||||||
|
let loadingTask = this.loadingTask;
|
||||||
|
if (loadingTask.onProgress) {
|
||||||
|
loadingTask.onProgress({
|
||||||
|
loaded: evt.loaded,
|
||||||
|
total: evt.total,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
headersCapability.resolve({
|
||||||
|
isStreamingSupported: fullReader.isStreamingSupported,
|
||||||
|
isRangeSupported: fullReader.isRangeSupported,
|
||||||
|
contentLength: fullReader.contentLength,
|
||||||
});
|
});
|
||||||
});
|
}, headersCapability.reject);
|
||||||
|
|
||||||
pdfDataRangeTransport.addProgressiveReadListener(function(chunk) {
|
return headersCapability.promise;
|
||||||
messageHandler.send('OnDataRange', {
|
}, this);
|
||||||
chunk,
|
|
||||||
|
messageHandler.on('GetRangeReader', function(data, sink) {
|
||||||
|
assert(this._networkStream);
|
||||||
|
let _rangeReader =
|
||||||
|
this._networkStream.getRangeReader(data.begin, data.end);
|
||||||
|
|
||||||
|
sink.onPull = () => {
|
||||||
|
_rangeReader.read().then(function({ value, done, }) {
|
||||||
|
if (done) {
|
||||||
|
sink.close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
assert(isArrayBuffer(value));
|
||||||
|
sink.enqueue(new Uint8Array(value), 1, [value]);
|
||||||
|
}).catch((reason) => {
|
||||||
|
sink.error(reason);
|
||||||
});
|
});
|
||||||
});
|
};
|
||||||
|
|
||||||
messageHandler.on('RequestDataRange',
|
sink.onCancel = (reason) => {
|
||||||
function transportDataRange(data) {
|
_rangeReader.cancel(reason);
|
||||||
pdfDataRangeTransport.requestDataRange(data.begin, data.end);
|
};
|
||||||
}, this);
|
}, this);
|
||||||
}
|
|
||||||
|
|
||||||
messageHandler.on('GetDoc', function transportDoc(data) {
|
messageHandler.on('GetDoc', function transportDoc(data) {
|
||||||
var pdfInfo = data.pdfInfo;
|
var pdfInfo = data.pdfInfo;
|
||||||
@ -1668,9 +1753,6 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
}, this);
|
}, this);
|
||||||
|
|
||||||
messageHandler.on('PDFManagerReady', function transportPage(data) {
|
messageHandler.on('PDFManagerReady', function transportPage(data) {
|
||||||
if (this.pdfDataRangeTransport) {
|
|
||||||
this.pdfDataRangeTransport.transportReady();
|
|
||||||
}
|
|
||||||
}, this);
|
}, this);
|
||||||
|
|
||||||
messageHandler.on('StartRenderPage', function transportRender(data) {
|
messageHandler.on('StartRenderPage', function transportRender(data) {
|
||||||
@ -2335,6 +2417,7 @@ export {
|
|||||||
PDFWorker,
|
PDFWorker,
|
||||||
PDFDocumentProxy,
|
PDFDocumentProxy,
|
||||||
PDFPageProxy,
|
PDFPageProxy,
|
||||||
|
setPDFNetworkStreamClass,
|
||||||
_UnsupportedManager,
|
_UnsupportedManager,
|
||||||
version,
|
version,
|
||||||
build,
|
build,
|
||||||
|
@ -17,7 +17,7 @@ import {
|
|||||||
assert, createPromiseCapability, globalScope, isInt, MissingPDFException,
|
assert, createPromiseCapability, globalScope, isInt, MissingPDFException,
|
||||||
UnexpectedResponseException
|
UnexpectedResponseException
|
||||||
} from '../shared/util';
|
} from '../shared/util';
|
||||||
import { setPDFNetworkStreamClass } from './worker';
|
import { setPDFNetworkStreamClass } from './api';
|
||||||
|
|
||||||
if (typeof PDFJSDev !== 'undefined' && PDFJSDev.test('FIREFOX || MOZCENTRAL')) {
|
if (typeof PDFJSDev !== 'undefined' && PDFJSDev.test('FIREFOX || MOZCENTRAL')) {
|
||||||
throw new Error('Module "./network" shall not ' +
|
throw new Error('Module "./network" shall not ' +
|
243
src/display/transport_stream.js
Normal file
243
src/display/transport_stream.js
Normal file
@ -0,0 +1,243 @@
|
|||||||
|
/* Copyright 2012 Mozilla Foundation
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { assert, createPromiseCapability } from '../shared/util';
|
||||||
|
|
||||||
|
/** @implements {IPDFStream} */
|
||||||
|
var PDFDataTransportStream = (function PDFDataTransportStreamClosure() {
|
||||||
|
function PDFDataTransportStream(params, pdfDataRangeTransport) {
|
||||||
|
assert(pdfDataRangeTransport);
|
||||||
|
|
||||||
|
this._queuedChunks = [];
|
||||||
|
var initialData = params.initialData;
|
||||||
|
if (initialData && initialData.length > 0) {
|
||||||
|
let buffer = new Uint8Array(initialData).buffer;
|
||||||
|
this._queuedChunks.push(buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
this._pdfDataRangeTransport = pdfDataRangeTransport;
|
||||||
|
this._isRangeSupported = !(params.disableRange);
|
||||||
|
this._isStreamingSupported = !(params.disableStream);
|
||||||
|
this._contentLength = params.length;
|
||||||
|
|
||||||
|
this._fullRequestReader = null;
|
||||||
|
this._rangeReaders = [];
|
||||||
|
|
||||||
|
this._pdfDataRangeTransport.addRangeListener((begin, chunk) => {
|
||||||
|
this._onReceiveData({ begin, chunk, });
|
||||||
|
});
|
||||||
|
|
||||||
|
this._pdfDataRangeTransport.addProgressListener((loaded) => {
|
||||||
|
this._onProgress({ loaded, });
|
||||||
|
});
|
||||||
|
|
||||||
|
this._pdfDataRangeTransport.addProgressiveReadListener((chunk) => {
|
||||||
|
this._onReceiveData({ chunk, });
|
||||||
|
});
|
||||||
|
|
||||||
|
this._pdfDataRangeTransport.transportReady();
|
||||||
|
}
|
||||||
|
PDFDataTransportStream.prototype = {
|
||||||
|
_onReceiveData: function PDFDataTransportStream_onReceiveData(args) {
|
||||||
|
let buffer = new Uint8Array(args.chunk).buffer;
|
||||||
|
if (args.begin === undefined) {
|
||||||
|
if (this._fullRequestReader) {
|
||||||
|
this._fullRequestReader._enqueue(buffer);
|
||||||
|
} else {
|
||||||
|
this._queuedChunks.push(buffer);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var found = this._rangeReaders.some(function (rangeReader) {
|
||||||
|
if (rangeReader._begin !== args.begin) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
rangeReader._enqueue(buffer);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
assert(found);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_onProgress: function PDFDataTransportStream_onDataProgress(evt) {
|
||||||
|
if (this._rangeReaders.length > 0) {
|
||||||
|
// Reporting to first range reader.
|
||||||
|
var firstReader = this._rangeReaders[0];
|
||||||
|
if (firstReader.onProgress) {
|
||||||
|
firstReader.onProgress({ loaded: evt.loaded, });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_removeRangeReader:
|
||||||
|
function PDFDataTransportStream_removeRangeReader(reader) {
|
||||||
|
var i = this._rangeReaders.indexOf(reader);
|
||||||
|
if (i >= 0) {
|
||||||
|
this._rangeReaders.splice(i, 1);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getFullReader: function PDFDataTransportStream_getFullReader() {
|
||||||
|
assert(!this._fullRequestReader);
|
||||||
|
var queuedChunks = this._queuedChunks;
|
||||||
|
this._queuedChunks = null;
|
||||||
|
return new PDFDataTransportStreamReader(this, queuedChunks);
|
||||||
|
},
|
||||||
|
|
||||||
|
getRangeReader: function PDFDataTransportStream_getRangeReader(begin, end) {
|
||||||
|
var reader = new PDFDataTransportStreamRangeReader(this, begin, end);
|
||||||
|
this._pdfDataRangeTransport.requestDataRange(begin, end);
|
||||||
|
this._rangeReaders.push(reader);
|
||||||
|
return reader;
|
||||||
|
},
|
||||||
|
|
||||||
|
cancelAllRequests:
|
||||||
|
function PDFDataTransportStream_cancelAllRequests(reason) {
|
||||||
|
if (this._fullRequestReader) {
|
||||||
|
this._fullRequestReader.cancel(reason);
|
||||||
|
}
|
||||||
|
var readers = this._rangeReaders.slice(0);
|
||||||
|
readers.forEach(function (rangeReader) {
|
||||||
|
rangeReader.cancel(reason);
|
||||||
|
});
|
||||||
|
this._pdfDataRangeTransport.abort();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/** @implements {IPDFStreamReader} */
|
||||||
|
function PDFDataTransportStreamReader(stream, queuedChunks) {
|
||||||
|
this._stream = stream;
|
||||||
|
this._done = false;
|
||||||
|
this._queuedChunks = queuedChunks || [];
|
||||||
|
this._requests = [];
|
||||||
|
this._headersReady = Promise.resolve();
|
||||||
|
stream._fullRequestReader = this;
|
||||||
|
|
||||||
|
this.onProgress = null; // not used
|
||||||
|
}
|
||||||
|
PDFDataTransportStreamReader.prototype = {
|
||||||
|
_enqueue: function PDFDataTransportStreamReader_enqueue(chunk) {
|
||||||
|
if (this._done) {
|
||||||
|
return; // ignore new data
|
||||||
|
}
|
||||||
|
if (this._requests.length > 0) {
|
||||||
|
var requestCapability = this._requests.shift();
|
||||||
|
requestCapability.resolve({ value: chunk, done: false, });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._queuedChunks.push(chunk);
|
||||||
|
},
|
||||||
|
|
||||||
|
get headersReady() {
|
||||||
|
return this._headersReady;
|
||||||
|
},
|
||||||
|
|
||||||
|
get isRangeSupported() {
|
||||||
|
return this._stream._isRangeSupported;
|
||||||
|
},
|
||||||
|
|
||||||
|
get isStreamingSupported() {
|
||||||
|
return this._stream._isStreamingSupported;
|
||||||
|
},
|
||||||
|
|
||||||
|
get contentLength() {
|
||||||
|
return this._stream._contentLength;
|
||||||
|
},
|
||||||
|
|
||||||
|
read: function PDFDataTransportStreamReader_read() {
|
||||||
|
if (this._queuedChunks.length > 0) {
|
||||||
|
var chunk = this._queuedChunks.shift();
|
||||||
|
return Promise.resolve({ value: chunk, done: false, });
|
||||||
|
}
|
||||||
|
if (this._done) {
|
||||||
|
return Promise.resolve({ value: undefined, done: true, });
|
||||||
|
}
|
||||||
|
var requestCapability = createPromiseCapability();
|
||||||
|
this._requests.push(requestCapability);
|
||||||
|
return requestCapability.promise;
|
||||||
|
},
|
||||||
|
|
||||||
|
cancel: function PDFDataTransportStreamReader_cancel(reason) {
|
||||||
|
this._done = true;
|
||||||
|
this._requests.forEach(function (requestCapability) {
|
||||||
|
requestCapability.resolve({ value: undefined, done: true, });
|
||||||
|
});
|
||||||
|
this._requests = [];
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/** @implements {IPDFStreamRangeReader} */
|
||||||
|
function PDFDataTransportStreamRangeReader(stream, begin, end) {
|
||||||
|
this._stream = stream;
|
||||||
|
this._begin = begin;
|
||||||
|
this._end = end;
|
||||||
|
this._queuedChunk = null;
|
||||||
|
this._requests = [];
|
||||||
|
this._done = false;
|
||||||
|
|
||||||
|
this.onProgress = null;
|
||||||
|
}
|
||||||
|
PDFDataTransportStreamRangeReader.prototype = {
|
||||||
|
_enqueue: function PDFDataTransportStreamRangeReader_enqueue(chunk) {
|
||||||
|
if (this._done) {
|
||||||
|
return; // ignore new data
|
||||||
|
}
|
||||||
|
if (this._requests.length === 0) {
|
||||||
|
this._queuedChunk = chunk;
|
||||||
|
} else {
|
||||||
|
var requestsCapability = this._requests.shift();
|
||||||
|
requestsCapability.resolve({ value: chunk, done: false, });
|
||||||
|
this._requests.forEach(function (requestCapability) {
|
||||||
|
requestCapability.resolve({ value: undefined, done: true, });
|
||||||
|
});
|
||||||
|
this._requests = [];
|
||||||
|
}
|
||||||
|
this._done = true;
|
||||||
|
this._stream._removeRangeReader(this);
|
||||||
|
},
|
||||||
|
|
||||||
|
get isStreamingSupported() {
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
read: function PDFDataTransportStreamRangeReader_read() {
|
||||||
|
if (this._queuedChunk) {
|
||||||
|
let chunk = this._queuedChunk;
|
||||||
|
this._queuedChunk = null;
|
||||||
|
return Promise.resolve({ value: chunk, done: false, });
|
||||||
|
}
|
||||||
|
if (this._done) {
|
||||||
|
return Promise.resolve({ value: undefined, done: true, });
|
||||||
|
}
|
||||||
|
var requestCapability = createPromiseCapability();
|
||||||
|
this._requests.push(requestCapability);
|
||||||
|
return requestCapability.promise;
|
||||||
|
},
|
||||||
|
|
||||||
|
cancel: function PDFDataTransportStreamRangeReader_cancel(reason) {
|
||||||
|
this._done = true;
|
||||||
|
this._requests.forEach(function (requestCapability) {
|
||||||
|
requestCapability.resolve({ value: undefined, done: true, });
|
||||||
|
});
|
||||||
|
this._requests = [];
|
||||||
|
this._stream._removeRangeReader(this);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return PDFDataTransportStream;
|
||||||
|
})();
|
||||||
|
|
||||||
|
export {
|
||||||
|
PDFDataTransportStream,
|
||||||
|
};
|
@ -29,6 +29,11 @@ var pdfjsDisplayAnnotationLayer = require('./display/annotation_layer.js');
|
|||||||
var pdfjsDisplayDOMUtils = require('./display/dom_utils.js');
|
var pdfjsDisplayDOMUtils = require('./display/dom_utils.js');
|
||||||
var pdfjsDisplaySVG = require('./display/svg.js');
|
var pdfjsDisplaySVG = require('./display/svg.js');
|
||||||
|
|
||||||
|
if (typeof PDFJSDev === 'undefined' ||
|
||||||
|
!PDFJSDev.test('FIREFOX || MOZCENTRAL')) {
|
||||||
|
require('./display/network.js');
|
||||||
|
}
|
||||||
|
|
||||||
exports.PDFJS = pdfjsDisplayGlobal.PDFJS;
|
exports.PDFJS = pdfjsDisplayGlobal.PDFJS;
|
||||||
exports.build = pdfjsDisplayAPI.build;
|
exports.build = pdfjsDisplayAPI.build;
|
||||||
exports.version = pdfjsDisplayAPI.version;
|
exports.version = pdfjsDisplayAPI.version;
|
||||||
|
5
src/pdf.worker.js
vendored
5
src/pdf.worker.js
vendored
@ -21,9 +21,4 @@ var pdfjsBuild = PDFJSDev.eval('BUNDLE_BUILD');
|
|||||||
|
|
||||||
var pdfjsCoreWorker = require('./core/worker.js');
|
var pdfjsCoreWorker = require('./core/worker.js');
|
||||||
|
|
||||||
if (typeof PDFJSDev === 'undefined' ||
|
|
||||||
!PDFJSDev.test('FIREFOX || MOZCENTRAL')) {
|
|
||||||
require('./core/network.js');
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.WorkerMessageHandler = pdfjsCoreWorker.WorkerMessageHandler;
|
exports.WorkerMessageHandler = pdfjsCoreWorker.WorkerMessageHandler;
|
||||||
|
@ -1222,6 +1222,20 @@ function resolveCall(fn, args, thisArg = null) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function wrapReason(reason) {
|
||||||
|
if (typeof reason !== 'object') {
|
||||||
|
return reason;
|
||||||
|
}
|
||||||
|
switch (reason.name) {
|
||||||
|
case 'MissingPDFException':
|
||||||
|
return new MissingPDFException(reason.message);
|
||||||
|
case 'UnexpectedResponseException':
|
||||||
|
return new UnexpectedResponseException(reason.message, reason.status);
|
||||||
|
default:
|
||||||
|
return new UnknownErrorException(reason.message, reason.details);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function resolveOrReject(capability, success, reason) {
|
function resolveOrReject(capability, success, reason) {
|
||||||
if (success) {
|
if (success) {
|
||||||
capability.resolve();
|
capability.resolve();
|
||||||
@ -1431,13 +1445,14 @@ MessageHandler.prototype = {
|
|||||||
let targetName = data.sourceName;
|
let targetName = data.sourceName;
|
||||||
let capability = createPromiseCapability();
|
let capability = createPromiseCapability();
|
||||||
|
|
||||||
let sendStreamRequest = ({ stream, chunk, success, reason, }) => {
|
let sendStreamRequest = ({ stream, chunk, transfers,
|
||||||
this.comObj.postMessage({ sourceName, targetName, stream, streamId,
|
success, reason, }) => {
|
||||||
chunk, success, reason, });
|
this.postMessage({ sourceName, targetName, stream, streamId,
|
||||||
|
chunk, success, reason, }, transfers);
|
||||||
};
|
};
|
||||||
|
|
||||||
let streamSink = {
|
let streamSink = {
|
||||||
enqueue(chunk, size = 1) {
|
enqueue(chunk, size = 1, transfers) {
|
||||||
if (this.isCancelled) {
|
if (this.isCancelled) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1450,7 +1465,7 @@ MessageHandler.prototype = {
|
|||||||
this.sinkCapability = createPromiseCapability();
|
this.sinkCapability = createPromiseCapability();
|
||||||
this.ready = this.sinkCapability.promise;
|
this.ready = this.sinkCapability.promise;
|
||||||
}
|
}
|
||||||
sendStreamRequest({ stream: 'enqueue', chunk, });
|
sendStreamRequest({ stream: 'enqueue', chunk, transfers, });
|
||||||
},
|
},
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
@ -1462,6 +1477,10 @@ MessageHandler.prototype = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
error(reason) {
|
error(reason) {
|
||||||
|
if (this.isCancelled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.isCancelled = true;
|
||||||
sendStreamRequest({ stream: 'error', reason, });
|
sendStreamRequest({ stream: 'error', reason, });
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -1510,11 +1529,11 @@ MessageHandler.prototype = {
|
|||||||
switch (data.stream) {
|
switch (data.stream) {
|
||||||
case 'start_complete':
|
case 'start_complete':
|
||||||
resolveOrReject(this.streamControllers[data.streamId].startCall,
|
resolveOrReject(this.streamControllers[data.streamId].startCall,
|
||||||
data.success, data.reason);
|
data.success, wrapReason(data.reason));
|
||||||
break;
|
break;
|
||||||
case 'pull_complete':
|
case 'pull_complete':
|
||||||
resolveOrReject(this.streamControllers[data.streamId].pullCall,
|
resolveOrReject(this.streamControllers[data.streamId].pullCall,
|
||||||
data.success, data.reason);
|
data.success, wrapReason(data.reason));
|
||||||
break;
|
break;
|
||||||
case 'pull':
|
case 'pull':
|
||||||
// Ignore any pull after close is called.
|
// Ignore any pull after close is called.
|
||||||
@ -1539,11 +1558,15 @@ MessageHandler.prototype = {
|
|||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
case 'enqueue':
|
case 'enqueue':
|
||||||
|
assert(this.streamControllers[data.streamId],
|
||||||
|
'enqueue should have stream controller');
|
||||||
if (!this.streamControllers[data.streamId].isClosed) {
|
if (!this.streamControllers[data.streamId].isClosed) {
|
||||||
this.streamControllers[data.streamId].controller.enqueue(data.chunk);
|
this.streamControllers[data.streamId].controller.enqueue(data.chunk);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'close':
|
case 'close':
|
||||||
|
assert(this.streamControllers[data.streamId],
|
||||||
|
'close should have stream controller');
|
||||||
if (this.streamControllers[data.streamId].isClosed) {
|
if (this.streamControllers[data.streamId].isClosed) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1552,12 +1575,15 @@ MessageHandler.prototype = {
|
|||||||
deleteStreamController();
|
deleteStreamController();
|
||||||
break;
|
break;
|
||||||
case 'error':
|
case 'error':
|
||||||
this.streamControllers[data.streamId].controller.error(data.reason);
|
assert(this.streamControllers[data.streamId],
|
||||||
|
'error should have stream controller');
|
||||||
|
this.streamControllers[data.streamId].controller.
|
||||||
|
error(wrapReason(data.reason));
|
||||||
deleteStreamController();
|
deleteStreamController();
|
||||||
break;
|
break;
|
||||||
case 'cancel_complete':
|
case 'cancel_complete':
|
||||||
resolveOrReject(this.streamControllers[data.streamId].cancelCall,
|
resolveOrReject(this.streamControllers[data.streamId].cancelCall,
|
||||||
data.success, data.reason);
|
data.success, wrapReason(data.reason));
|
||||||
deleteStreamController();
|
deleteStreamController();
|
||||||
break;
|
break;
|
||||||
case 'cancel':
|
case 'cancel':
|
||||||
@ -1565,13 +1591,14 @@ MessageHandler.prototype = {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
resolveCall(this.streamSinks[data.streamId].onCancel,
|
resolveCall(this.streamSinks[data.streamId].onCancel,
|
||||||
[data.reason]).then(() => {
|
[wrapReason(data.reason)]).then(() => {
|
||||||
sendStreamResponse({ stream: 'cancel_complete', success: true, });
|
sendStreamResponse({ stream: 'cancel_complete', success: true, });
|
||||||
}, (reason) => {
|
}, (reason) => {
|
||||||
sendStreamResponse({ stream: 'cancel_complete',
|
sendStreamResponse({ stream: 'cancel_complete',
|
||||||
success: false, reason, });
|
success: false, reason, });
|
||||||
});
|
});
|
||||||
this.streamSinks[data.streamId].sinkCapability.reject(data.reason);
|
this.streamSinks[data.streamId].sinkCapability.
|
||||||
|
reject(wrapReason(data.reason));
|
||||||
this.streamSinks[data.streamId].isCancelled = true;
|
this.streamSinks[data.streamId].isCancelled = true;
|
||||||
delete this.streamSinks[data.streamId];
|
delete this.streamSinks[data.streamId];
|
||||||
break;
|
break;
|
||||||
|
@ -28,7 +28,6 @@ importScripts('./shared/compatibility.js');
|
|||||||
importScripts('../node_modules/systemjs/dist/system.js');
|
importScripts('../node_modules/systemjs/dist/system.js');
|
||||||
importScripts('../systemjs.config.js');
|
importScripts('../systemjs.config.js');
|
||||||
|
|
||||||
Promise.all([SystemJS.import('pdfjs/core/network'),
|
SystemJS.import('pdfjs/core/worker').then(function () {
|
||||||
SystemJS.import('pdfjs/core/worker')]).then(function () {
|
|
||||||
// Worker is loaded at this point.
|
// Worker is loaded at this point.
|
||||||
});
|
});
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { PDFNetworkStream } from '../../src/core/network';
|
import { PDFNetworkStream } from '../../src/display/network';
|
||||||
|
|
||||||
describe('network', function() {
|
describe('network', function() {
|
||||||
var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
|
var pdf1 = new URL('../pdfs/tracemonkey.pdf', window.location).href;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user