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…
Reference in New Issue
Block a user