Merge pull request #8321 from Snuffleupagus/issue-8263
Change `getPDFFileNameFromURL` to ignore `data:` URLs for performance reasons (issue 8263)
This commit is contained in:
		
						commit
						c44fd3d6e2
					
				| @ -1190,7 +1190,7 @@ var createBlob = function createBlob(data, contentType) { | |||||||
|   if (typeof Blob !== 'undefined') { |   if (typeof Blob !== 'undefined') { | ||||||
|     return new Blob([data], { type: contentType }); |     return new Blob([data], { type: contentType }); | ||||||
|   } |   } | ||||||
|   warn('The "Blob" constructor is not supported.'); |   throw new Error('The "Blob" constructor is not supported.'); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| var createObjectURL = (function createObjectURLClosure() { | var createObjectURL = (function createObjectURLClosure() { | ||||||
| @ -1198,9 +1198,8 @@ var createObjectURL = (function createObjectURLClosure() { | |||||||
|   var digits = |   var digits = | ||||||
|     'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; |     'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; | ||||||
| 
 | 
 | ||||||
|   return function createObjectURL(data, contentType, forceDataSchema) { |   return function createObjectURL(data, contentType, forceDataSchema = false) { | ||||||
|     if (!forceDataSchema && |     if (!forceDataSchema) { | ||||||
|         typeof URL !== 'undefined' && URL.createObjectURL) { |  | ||||||
|       var blob = createBlob(data, contentType); |       var blob = createBlob(data, contentType); | ||||||
|       return URL.createObjectURL(blob); |       return URL.createObjectURL(blob); | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -17,16 +17,20 @@ | |||||||
| (function (root, factory) { | (function (root, factory) { | ||||||
|   if (typeof define === 'function' && define.amd) { |   if (typeof define === 'function' && define.amd) { | ||||||
|     define('pdfjs-test/unit/ui_utils_spec', ['exports', |     define('pdfjs-test/unit/ui_utils_spec', ['exports', | ||||||
|            'pdfjs-web/ui_utils'], factory); |       'pdfjs-web/ui_utils', 'pdfjs/shared/util'], factory); | ||||||
|   } else if (typeof exports !== 'undefined') { |   } else if (typeof exports !== 'undefined') { | ||||||
|     factory(exports, require('../../web/ui_utils.js')); |     factory(exports, require('../../web/ui_utils.js'), | ||||||
|  |       require('../../src/shared/util.js')); | ||||||
|   } else { |   } else { | ||||||
|     factory((root.pdfjsTestUnitUiUtilsSpec = {}), root.pdfjsWebUiUtils); |     factory((root.pdfjsTestUnitUiUtilsSpec = {}), root.pdfjsWebUiUtils, | ||||||
|  |       root.pdfjsSharedUtil); | ||||||
|   } |   } | ||||||
| }(this, function (exports, webUiUtils) { | }(this, function (exports, webUiUtils, sharedUtil) { | ||||||
| 
 | 
 | ||||||
| var binarySearchFirstItem = webUiUtils.binarySearchFirstItem; | var binarySearchFirstItem = webUiUtils.binarySearchFirstItem; | ||||||
|  | var getPDFFileNameFromURL = webUiUtils.getPDFFileNameFromURL; | ||||||
| var EventBus = webUiUtils.EventBus; | var EventBus = webUiUtils.EventBus; | ||||||
|  | var createObjectURL = sharedUtil.createObjectURL; | ||||||
| 
 | 
 | ||||||
| describe('ui_utils', function() { | describe('ui_utils', function() { | ||||||
|   describe('binary search', function() { |   describe('binary search', function() { | ||||||
| @ -57,6 +61,130 @@ describe('ui_utils', function() { | |||||||
|     }); |     }); | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|  |   describe('getPDFFileNameFromURL', function() { | ||||||
|  |     it('gets PDF filename', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/file1.pdf')).toEqual('file1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         'http://www.example.com/pdfs/file2.pdf')).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets fallback filename', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/file1.txt')).toEqual('document.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         'http://www.example.com/pdfs/file2.txt')).toEqual('document.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets custom fallback filename', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/file1.txt', 'qwerty1.pdf')). | ||||||
|  |         toEqual('qwerty1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL('http://www.example.com/pdfs/file2.txt', | ||||||
|  |         'qwerty2.pdf')).toEqual('qwerty2.pdf'); | ||||||
|  | 
 | ||||||
|  |       // An empty string should be a valid custom fallback filename.
 | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/file3.txt', '')).toEqual(''); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from URL containing leading/trailing whitespace', | ||||||
|  |         function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         '   /pdfs/file1.pdf   ')).toEqual('file1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         '   http://www.example.com/pdfs/file2.pdf   ')).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from query string', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         '/pdfs/pdfs.html?name=file1.pdf')).toEqual('file1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         'http://www.example.com/pdfs/pdf.html?file2.pdf')).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from hash string', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         '/pdfs/pdfs.html#name=file1.pdf')).toEqual('file1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         'http://www.example.com/pdfs/pdf.html#file2.pdf')).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets correct PDF filename when multiple ones are present', function() { | ||||||
|  |       // Relative URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         '/pdfs/file1.pdf?name=file.pdf')).toEqual('file1.pdf'); | ||||||
|  |       // Absolute URL
 | ||||||
|  |       expect(getPDFFileNameFromURL( | ||||||
|  |         'http://www.example.com/pdfs/file2.pdf#file.pdf')).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from URI-encoded data', function() { | ||||||
|  |       var encodedUrl = encodeURIComponent( | ||||||
|  |         'http://www.example.com/pdfs/file1.pdf'); | ||||||
|  |       expect(getPDFFileNameFromURL(encodedUrl)).toEqual('file1.pdf'); | ||||||
|  | 
 | ||||||
|  |       var encodedUrlWithQuery = encodeURIComponent( | ||||||
|  |         'http://www.example.com/pdfs/file.txt?file2.pdf'); | ||||||
|  |       expect(getPDFFileNameFromURL(encodedUrlWithQuery)).toEqual('file2.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from data mistaken for URI-encoded', function() { | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/%AA.pdf')).toEqual('%AA.pdf'); | ||||||
|  | 
 | ||||||
|  |       expect(getPDFFileNameFromURL('/pdfs/%2F.pdf')).toEqual('%2F.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from (some) standard protocols', function() { | ||||||
|  |       // HTTP
 | ||||||
|  |       expect(getPDFFileNameFromURL('http://www.example.com/file1.pdf')). | ||||||
|  |         toEqual('file1.pdf'); | ||||||
|  |       // HTTPS
 | ||||||
|  |       expect(getPDFFileNameFromURL('https://www.example.com/file2.pdf')). | ||||||
|  |         toEqual('file2.pdf'); | ||||||
|  |       // File
 | ||||||
|  |       expect(getPDFFileNameFromURL('file:///path/to/files/file3.pdf')). | ||||||
|  |         toEqual('file3.pdf'); | ||||||
|  |       // FTP
 | ||||||
|  |       expect(getPDFFileNameFromURL('ftp://www.example.com/file4.pdf')). | ||||||
|  |         toEqual('file4.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets PDF filename from query string appended to "blob:" URL', | ||||||
|  |         function() { | ||||||
|  |       var typedArray = new Uint8Array([1, 2, 3, 4, 5]); | ||||||
|  |       var blobUrl = createObjectURL(typedArray, 'application/pdf'); | ||||||
|  |       // Sanity check to ensure that a "blob:" URL was returned.
 | ||||||
|  |       expect(blobUrl.indexOf('blob:') === 0).toEqual(true); | ||||||
|  | 
 | ||||||
|  |       expect(getPDFFileNameFromURL(blobUrl + '?file.pdf')).toEqual('file.pdf'); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('gets fallback filename from query string appended to "data:" URL', | ||||||
|  |         function() { | ||||||
|  |       var typedArray = new Uint8Array([1, 2, 3, 4, 5]); | ||||||
|  |       var dataUrl = createObjectURL(typedArray, 'application/pdf', | ||||||
|  |                                     /* forceDataSchema = */ true); | ||||||
|  |       // Sanity check to ensure that a "data:" URL was returned.
 | ||||||
|  |       expect(dataUrl.indexOf('data:') === 0).toEqual(true); | ||||||
|  | 
 | ||||||
|  |       expect(getPDFFileNameFromURL(dataUrl + '?file1.pdf')). | ||||||
|  |         toEqual('document.pdf'); | ||||||
|  | 
 | ||||||
|  |       // Should correctly detect a "data:" URL with leading whitespace.
 | ||||||
|  |       expect(getPDFFileNameFromURL('     ' + dataUrl + '?file2.pdf')). | ||||||
|  |         toEqual('document.pdf'); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|   describe('EventBus', function () { |   describe('EventBus', function () { | ||||||
|     it('dispatch event', function () { |     it('dispatch event', function () { | ||||||
|       var eventBus = new EventBus(); |       var eventBus = new EventBus(); | ||||||
|  | |||||||
| @ -79,13 +79,15 @@ var PDFAttachmentViewer = (function PDFAttachmentViewerClosure() { | |||||||
|     /** |     /** | ||||||
|      * @private |      * @private | ||||||
|      */ |      */ | ||||||
|     _bindPdfLink: |     _bindPdfLink(button, content, filename) { | ||||||
|         function PDFAttachmentViewer_bindPdfLink(button, content, filename) { |       if (PDFJS.disableCreateObjectURL) { | ||||||
|  |         throw new Error('bindPdfLink: ' + | ||||||
|  |                         'Unsupported "PDFJS.disableCreateObjectURL" value.'); | ||||||
|  |       } | ||||||
|       var blobUrl; |       var blobUrl; | ||||||
|       button.onclick = function() { |       button.onclick = function() { | ||||||
|         if (!blobUrl) { |         if (!blobUrl) { | ||||||
|           blobUrl = createObjectURL( |           blobUrl = createObjectURL(content, 'application/pdf'); | ||||||
|             content, 'application/pdf', PDFJS.disableCreateObjectURL); |  | ||||||
|         } |         } | ||||||
|         var viewerUrl; |         var viewerUrl; | ||||||
|         if (typeof PDFJSDev === 'undefined' || PDFJSDev.test('GENERIC')) { |         if (typeof PDFJSDev === 'undefined' || PDFJSDev.test('GENERIC')) { | ||||||
| @ -97,10 +99,8 @@ var PDFAttachmentViewer = (function PDFAttachmentViewerClosure() { | |||||||
|           // eslint-disable-next-line no-undef
 |           // eslint-disable-next-line no-undef
 | ||||||
|           viewerUrl = chrome.runtime.getURL('/content/web/viewer.html') + |           viewerUrl = chrome.runtime.getURL('/content/web/viewer.html') + | ||||||
|             '?file=' + encodeURIComponent(blobUrl + '#' + filename); |             '?file=' + encodeURIComponent(blobUrl + '#' + filename); | ||||||
|         } else { |         } else if (PDFJSDev.test('FIREFOX || MOZCENTRAL')) { | ||||||
|           // Let Firefox's content handler catch the URL and display the PDF.
 |           // Let Firefox's content handler catch the URL and display the PDF.
 | ||||||
|           // In Firefox PDFJS.disableCreateObjectURL is always false, so
 |  | ||||||
|           // blobUrl is always a blob:-URL and never a data:-URL.
 |  | ||||||
|           viewerUrl = blobUrl + '?' + encodeURIComponent(filename); |           viewerUrl = blobUrl + '?' + encodeURIComponent(filename); | ||||||
|         } |         } | ||||||
|         window.open(viewerUrl); |         window.open(viewerUrl); | ||||||
| @ -151,7 +151,7 @@ var PDFAttachmentViewer = (function PDFAttachmentViewerClosure() { | |||||||
|         div.className = 'attachmentsItem'; |         div.className = 'attachmentsItem'; | ||||||
|         var button = document.createElement('button'); |         var button = document.createElement('button'); | ||||||
|         button.textContent = filename; |         button.textContent = filename; | ||||||
|         if (/\.pdf$/i.test(filename)) { |         if (/\.pdf$/i.test(filename) && !PDFJS.disableCreateObjectURL) { | ||||||
|           this._bindPdfLink(button, item.content, filename); |           this._bindPdfLink(button, item.content, filename); | ||||||
|         } else { |         } else { | ||||||
|           this._bindLink(button, item.content, filename); |           this._bindLink(button, item.content, filename); | ||||||
|  | |||||||
| @ -353,15 +353,26 @@ function noContextMenuHandler(e) { | |||||||
|   e.preventDefault(); |   e.preventDefault(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | function isDataSchema(url) { | ||||||
|  |   var i = 0, ii = url.length; | ||||||
|  |   while (i < ii && url[i].trim() === '') { | ||||||
|  |     i++; | ||||||
|  |   } | ||||||
|  |   return url.substr(i, 5).toLowerCase() === 'data:'; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /** | /** | ||||||
|  * Returns the filename or guessed filename from the url (see issue 3455). |  * Returns the filename or guessed filename from the url (see issue 3455). | ||||||
|  * url {String} The original PDF location. |  * @param {string} url - The original PDF location. | ||||||
|  * defaultFilename {string} The value to return if the file name is unknown. |  * @param {string} defaultFilename - The value returned if the filename is | ||||||
|  * @return {String} Guessed PDF file name. |  *   unknown, or the protocol is unsupported. | ||||||
|  |  * @returns {string} Guessed PDF filename. | ||||||
|  */ |  */ | ||||||
| function getPDFFileNameFromURL(url, defaultFilename) { | function getPDFFileNameFromURL(url, defaultFilename = 'document.pdf') { | ||||||
|   if (typeof defaultFilename === 'undefined') { |   if (isDataSchema(url)) { | ||||||
|     defaultFilename = 'document.pdf'; |     console.warn('getPDFFileNameFromURL: ' + | ||||||
|  |                  'ignoring "data:" URL for performance reasons.'); | ||||||
|  |     return defaultFilename; | ||||||
|   } |   } | ||||||
|   var reURI = /^(?:(?:[^:]+:)?\/\/[^\/]+)?([^?#]*)(\?[^#]*)?(#.*)?$/; |   var reURI = /^(?:(?:[^:]+:)?\/\/[^\/]+)?([^?#]*)(\?[^#]*)?(#.*)?$/; | ||||||
|   //            SCHEME        HOST         1.PATH  2.QUERY   3.REF
 |   //            SCHEME        HOST         1.PATH  2.QUERY   3.REF
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user