Remove support for frames in old Chromium versions
Frames are only supported (via redirectUrl) in Chrome 35.0.1911.0+
This commit is contained in:
		
							parent
							
								
									69cf8c5fb3
								
							
						
					
					
						commit
						9394fc133c
					
				| @ -1,283 +0,0 @@ | |||||||
| /** |  | ||||||
|  * (c) 2013 Rob Wu <gwnRob@gmail.com> |  | ||||||
|  * Released under the MIT license |  | ||||||
|  * https://github.com/Rob--W/chrome-api/chrome.tabs.executeScriptInFrame
 |  | ||||||
|  * |  | ||||||
|  * Implements the chrome.tabs.executeScriptInFrame API. |  | ||||||
|  * This API is similar to the chrome.tabs.executeScript method, except |  | ||||||
|  * that it also recognizes the "frameId" property. |  | ||||||
|  * This frameId can be obtained through the webNavigation or webRequest API. |  | ||||||
|  * |  | ||||||
|  * When an error occurs, chrome.runtime.lastError is set. |  | ||||||
|  * |  | ||||||
|  * Required permissions: |  | ||||||
|  * webRequest |  | ||||||
|  * webRequestBlocking |  | ||||||
|  * Host permissions for the tab |  | ||||||
|  * |  | ||||||
|  * In addition, the following field must also be set in manifest.json: |  | ||||||
|  * "web_accessible_resources": ["getFrameId"] |  | ||||||
|  */ |  | ||||||
| /* globals chrome, console */ |  | ||||||
| 
 |  | ||||||
| (function() { |  | ||||||
|   /* jshint browser:true, maxlen:100 */ |  | ||||||
|   'use strict'; |  | ||||||
| 
 |  | ||||||
|   chrome.tabs.executeScriptInFrame = executeScript; |  | ||||||
| 
 |  | ||||||
|   // This URL is used to communicate the frameId. The resource is never
 |  | ||||||
|   // visited, so it should be a non-existent location. Do not use *, ", '
 |  | ||||||
|   // or line breaks in the file name.
 |  | ||||||
|   var URL_WHAT_IS_MY_FRAME_ID = chrome.extension.getURL('getFrameId'); |  | ||||||
|   // The callback will be called within ... ms:
 |  | ||||||
|   // Don't set a too low value.
 |  | ||||||
|   var MAXIMUM_RESPONSE_TIME_MS = 1000; |  | ||||||
| 
 |  | ||||||
|   // Callbacks are stored here until they're invoked.
 |  | ||||||
|   // Key = dummyUrl, value = callback function
 |  | ||||||
|   var callbacks = {}; |  | ||||||
| 
 |  | ||||||
|   chrome.webRequest.onBeforeRequest.addListener(function showFrameId(details) { |  | ||||||
|     // Positive integer frameId >= 0
 |  | ||||||
|     // Since an image is used as a data transport, we add 1 to get a
 |  | ||||||
|     // non-zero width.
 |  | ||||||
|     var frameId = details.frameId + 1; |  | ||||||
|     // Assume that the frameId fits in three bytes - which is a very
 |  | ||||||
|     // reasonable assumption.
 |  | ||||||
|     var width = String.fromCharCode(frameId & 0xFF, (frameId >> 8) & 0xFF); |  | ||||||
|     // When frameId > 0xFFFF, use the height to convey the additional
 |  | ||||||
|     // information. Again, add 1 to make sure that the height is non-zero.
 |  | ||||||
|     var height = String.fromCharCode((frameId >> 16) + 1, 0); |  | ||||||
|     // Convert data to base64 to avoid loss of bytes
 |  | ||||||
|     var image = 'data:image/gif;base64,' + btoa( |  | ||||||
|                 // 4749 4638 3961 (GIF header)
 |  | ||||||
|                 'GIF89a' + |  | ||||||
|                 // Logical Screen Width (LSB)
 |  | ||||||
|                 width + |  | ||||||
|                 // Logical Screen Height (LSB)
 |  | ||||||
|                 height + |  | ||||||
|                 // "No Global Color Table follows"
 |  | ||||||
|                 '\x00' + |  | ||||||
|                 // Background color
 |  | ||||||
|                 '\xff' + |  | ||||||
|                 // No aspect information is given
 |  | ||||||
|                 '\x00' + |  | ||||||
|                 // (image descriptor)
 |  | ||||||
|                 // Image Separator
 |  | ||||||
|                 '\x2c' + |  | ||||||
|                 // Image Position (Left & Top)
 |  | ||||||
|                 '\x00\x00\x00\x00' + |  | ||||||
|                 // Image Width (LSB)
 |  | ||||||
|                 width + |  | ||||||
|                 // Image Height (LSB)
 |  | ||||||
|                 height + |  | ||||||
|                 // Local Color Table is not present
 |  | ||||||
|                 '\x00' + |  | ||||||
|                 // (End of image descriptor)
 |  | ||||||
|                 // Image data
 |  | ||||||
|                 '\x02\x02\x44\x01\x00' + |  | ||||||
|                 // GIF trailer
 |  | ||||||
|                 '\x3b' |  | ||||||
|                 ); |  | ||||||
|     return {redirectUrl: image}; |  | ||||||
|   }, { |  | ||||||
|     urls: [URL_WHAT_IS_MY_FRAME_ID + '*'], |  | ||||||
|     types: ['image'] |  | ||||||
|   }, ['blocking']); |  | ||||||
| 
 |  | ||||||
|   chrome.runtime.onMessage.addListener(function(message, sender, |  | ||||||
|                                                 sendResponse) { |  | ||||||
|     if (message && message.executeScriptCallback) { |  | ||||||
|       var callback = callbacks[message.identifier]; |  | ||||||
|       if (callback) { |  | ||||||
|         if (message.hello) { |  | ||||||
|           clearTimeout(callback.timer); |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         delete callbacks[message.identifier]; |  | ||||||
|         // Result within an array to be consistent with the
 |  | ||||||
|         // chrome.tabs.executeScript API.
 |  | ||||||
|         callback([message.evalResult]); |  | ||||||
|       } else { |  | ||||||
|         console.warn('Callback not found for response in tab ' + |  | ||||||
|                      sender.tab.id); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * Execute content script in a specific frame. |  | ||||||
|    * |  | ||||||
|    * @param tabId {integer} required |  | ||||||
|    * @param details.frameId {integer} required |  | ||||||
|    * @param details.code {string} Code or file is required (not both) |  | ||||||
|    * @param details.file {string} Code or file is required (not both) |  | ||||||
|    * @param details.runAt {optional string} One of "document_start", |  | ||||||
|    *                                        "document_end", "document_idle" |  | ||||||
|    * @param callback {optional function(optional result array)} When an error |  | ||||||
|    *                                                            occurs, result |  | ||||||
|    *                                                            is not set. |  | ||||||
|    */ |  | ||||||
|   function executeScript(tabId, details, callback) { |  | ||||||
|     console.assert(typeof details === 'object', |  | ||||||
|                   'details must be an object (argument 0)'); |  | ||||||
|     var frameId = details.frameId; |  | ||||||
|     console.assert(typeof tabId === 'number', |  | ||||||
|                   'details.tabId must be a number'); |  | ||||||
|     console.assert(typeof frameId === 'number', |  | ||||||
|                   'details.frameId must be a number'); |  | ||||||
|     var sourceType = ('code' in details ? 'code' : 'file'); |  | ||||||
|     console.assert(sourceType in details, 'No source code or file specified'); |  | ||||||
|     var sourceValue = details[sourceType]; |  | ||||||
|     console.assert(typeof sourceValue === 'string', |  | ||||||
|                   'details.' + sourceType + ' must be a string'); |  | ||||||
|     var runAt = details.runAt; |  | ||||||
|     if (!callback) { |  | ||||||
|       callback = function() {/* no-op*/}; |  | ||||||
|     } |  | ||||||
|     console.assert(typeof callback === 'function', |  | ||||||
|                   'callback must be a function'); |  | ||||||
| 
 |  | ||||||
|     if (frameId === 0) { |  | ||||||
|       // No need for heavy lifting if we want to inject the script
 |  | ||||||
|       // in the main frame
 |  | ||||||
|       var injectDetails = { |  | ||||||
|         allFrames: false, |  | ||||||
|         runAt: runAt |  | ||||||
|       }; |  | ||||||
|       injectDetails[sourceType] = sourceValue; |  | ||||||
|       chrome.tabs.executeScript(tabId, injectDetails, callback); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     var identifier = Math.random().toString(36); |  | ||||||
| 
 |  | ||||||
|     if (sourceType === 'code') { |  | ||||||
|       executeScriptInFrame(); |  | ||||||
|     } else { // sourceType === 'file'
 |  | ||||||
|       (function() { |  | ||||||
|         var x = new XMLHttpRequest(); |  | ||||||
|         x.open('GET', chrome.extension.getURL(sourceValue), true); |  | ||||||
|         x.onload = function() { |  | ||||||
|           sourceValue = x.responseText; |  | ||||||
|           executeScriptInFrame(); |  | ||||||
|         }; |  | ||||||
|         x.onerror = function executeScriptResourceFetchError() { |  | ||||||
|           var message = 'Failed to load file: "' + sourceValue + '".'; |  | ||||||
|           console.error('executeScript: ' + message); |  | ||||||
|           chrome.runtime.lastError = chrome.extension.lastError = |  | ||||||
|             { message: message }; |  | ||||||
|           try { |  | ||||||
|             callback(); |  | ||||||
|           } finally { |  | ||||||
|             chrome.runtime.lastError = chrome.extension.lastError = undefined; |  | ||||||
|           } |  | ||||||
|         }; |  | ||||||
|         x.send(); |  | ||||||
|       })(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     function executeScriptInFrame() { |  | ||||||
|       callbacks[identifier] = callback; |  | ||||||
|       chrome.tabs.executeScript(tabId, { |  | ||||||
|         code: '(' + DETECT_FRAME + ')(' + |  | ||||||
|               'window,' + |  | ||||||
|                JSON.stringify(identifier) + ',' + |  | ||||||
|                frameId + ',' + |  | ||||||
|                JSON.stringify(sourceValue) + ')', |  | ||||||
|         allFrames: true, |  | ||||||
|         runAt: 'document_start' |  | ||||||
|       }, function(results) { |  | ||||||
|         if (results) { |  | ||||||
|           callback.timer = setTimeout(executeScriptTimedOut, |  | ||||||
|                                       MAXIMUM_RESPONSE_TIME_MS); |  | ||||||
|         } else { |  | ||||||
|           // Failed :(
 |  | ||||||
|           delete callbacks[identifier]; |  | ||||||
|           callback(); |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     function executeScriptTimedOut() { |  | ||||||
|       var callback = callbacks[identifier]; |  | ||||||
|       if (!callback) { |  | ||||||
|         return; |  | ||||||
|       } |  | ||||||
|       delete callbacks[identifier]; |  | ||||||
|       var message = 'Failed to execute script: Frame ' + frameId + |  | ||||||
|                     ' not found in tab ' + tabId; |  | ||||||
|       console.error('executeScript: ' + message); |  | ||||||
|       chrome.runtime.lastError = chrome.extension.lastError = |  | ||||||
|         { message: message }; |  | ||||||
|       try { |  | ||||||
|           callback(); |  | ||||||
|       } finally { |  | ||||||
|           chrome.runtime.lastError = chrome.extension.lastError = undefined; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * Code executed as a content script. |  | ||||||
|    */ |  | ||||||
|   var DETECT_FRAME = '' + function checkFrame(window, identifier, frameId, |  | ||||||
|                                               code) { |  | ||||||
|     var i; |  | ||||||
|     if ('__executeScript_frameId__' in window) { |  | ||||||
|       evalAsContentScript(); |  | ||||||
|     } else { |  | ||||||
|       // Do NOT use new Image() because of http://crbug.com/245296
 |  | ||||||
|       // in Chrome 27-29
 |  | ||||||
|       i = window.document.createElement('img'); |  | ||||||
|       i.onload = function() { |  | ||||||
|         window.__executeScript_frameId__ = (this.naturalWidth - 1) + |  | ||||||
|                                            (this.naturalHeight - 1 << 16); |  | ||||||
|         evalAsContentScript(); |  | ||||||
|       }; |  | ||||||
|       // Trigger webRequest event to get frameId
 |  | ||||||
|       // (append extra characters to bust the cache)
 |  | ||||||
|       i.src = 'URL_WHAT_IS_MY_FRAME_ID?' + |  | ||||||
|                Math.random().toString(36).slice(-6); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     for (i = 0 ; i < window.frames.length; ++i) { |  | ||||||
|       try { |  | ||||||
|         var frame = window.frames[i]; |  | ||||||
|         var scheme = frame.location.protocol; |  | ||||||
|         if (scheme !== 'https:' && scheme !== 'http:' && scheme !== 'file:') { |  | ||||||
|           checkFrame(frame, identifier, frameId, code); |  | ||||||
|         } |  | ||||||
|       } catch (e) { |  | ||||||
|         // blocked by same origin policy, so it's not a javascript:/about:blank
 |  | ||||||
|         // URL. chrome.tabs.executeScript will run the script for the frame.
 |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     function evalAsContentScript() { |  | ||||||
|       if (window.__executeScript_frameId__ !== frameId) { |  | ||||||
|         return; |  | ||||||
|       } |  | ||||||
|       // Send an early message to make sure that any blocking code
 |  | ||||||
|       // in the evaluated code does not cause the time-out in the background
 |  | ||||||
|       // page to be triggered
 |  | ||||||
|       chrome.runtime.sendMessage({ |  | ||||||
|         executeScriptCallback: true, |  | ||||||
|         hello: true, |  | ||||||
|         identifier: identifier |  | ||||||
|       }); |  | ||||||
|       var result = null; |  | ||||||
|       try { |  | ||||||
|         // jshint evil:true
 |  | ||||||
|         result = window.eval(code); |  | ||||||
|       } finally { |  | ||||||
|         chrome.runtime.sendMessage({ |  | ||||||
|           executeScriptCallback: true, |  | ||||||
|           evalResult: result, |  | ||||||
|           identifier: identifier |  | ||||||
|         }); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   }.toString().replace('URL_WHAT_IS_MY_FRAME_ID', URL_WHAT_IS_MY_FRAME_ID); |  | ||||||
| })(); |  | ||||||
| @ -57,7 +57,6 @@ | |||||||
|   }, |   }, | ||||||
|   "incognito": "split", |   "incognito": "split", | ||||||
|   "web_accessible_resources": [ |   "web_accessible_resources": [ | ||||||
|     "getFrameId", |  | ||||||
|     "content/web/viewer.html", |     "content/web/viewer.html", | ||||||
|     "http:/*", |     "http:/*", | ||||||
|     "https:/*", |     "https:/*", | ||||||
|  | |||||||
| @ -14,7 +14,6 @@ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||||||
| See the License for the specific language governing permissions and | See the License for the specific language governing permissions and | ||||||
| limitations under the License. | limitations under the License. | ||||||
| --> | --> | ||||||
| <script src="chrome.tabs.executeScriptInFrame.js"></script> |  | ||||||
| <script src="feature-detect.js"></script> | <script src="feature-detect.js"></script> | ||||||
| <script src="preserve-referer.js"></script> | <script src="preserve-referer.js"></script> | ||||||
| <script src="pdfHandler.js"></script> | <script src="pdfHandler.js"></script> | ||||||
|  | |||||||
| @ -134,45 +134,8 @@ chrome.webRequest.onHeadersReceived.addListener( | |||||||
|       }); |       }); | ||||||
|       return { cancel: true }; |       return { cancel: true }; | ||||||
|     } else { |     } else { | ||||||
|       // Sub frame. Requires some more work...
 |       console.warn('Child frames are not supported in ancient Chrome builds!'); | ||||||
|       // The navigation will be cancelled at the end of the webRequest cycle.
 |  | ||||||
|       chrome.webNavigation.onErrorOccurred.addListener(function listener(nav) { |  | ||||||
|         if (nav.tabId !== details.tabId || nav.frameId !== details.frameId) { |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         chrome.webNavigation.onErrorOccurred.removeListener(listener); |  | ||||||
| 
 |  | ||||||
|         // Locate frame and insert viewer
 |  | ||||||
|         chrome.tabs.executeScriptInFrame(details.tabId, { |  | ||||||
|           frameId: details.frameId, |  | ||||||
|           code: 'location.href = ' + JSON.stringify(viewerUrl) + ';' |  | ||||||
|         }, function(result) { |  | ||||||
|           if (!result) { |  | ||||||
|             console.warn('Frame not found! Opening viewer in new tab...'); |  | ||||||
|             chrome.tabs.create({ |  | ||||||
|               url: viewerUrl |  | ||||||
|             }); |  | ||||||
|           } |  | ||||||
|         }); |  | ||||||
|       }, { |  | ||||||
|         url: [{ urlEquals: details.url.split('#', 1)[0] }] |  | ||||||
|       }); |  | ||||||
|       // Prevent frame from rendering by using X-Frame-Options.
 |  | ||||||
|       // Do not use { cancel: true }, because that makes the frame inaccessible
 |  | ||||||
|       // to the content script that has to replace the frame's URL.
 |  | ||||||
|       return { |  | ||||||
|         responseHeaders: [{ |  | ||||||
|           name: 'X-Content-Type-Options', |  | ||||||
|           value: 'nosniff' |  | ||||||
|         }, { |  | ||||||
|           name: 'X-Frame-Options', |  | ||||||
|           value: 'deny' |  | ||||||
|         }] |  | ||||||
|       }; |  | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     // Immediately abort the request, because the frame that initiated the
 |  | ||||||
|     // request will be replaced with the PDF Viewer (within a split second).
 |  | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     urls: [ |     urls: [ | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user