4 changed files with 1 additions and 323 deletions
@ -1,283 +0,0 @@
@@ -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); |
||||
})(); |
Loading…
Reference in new issue