feat(extensions): add chrome.tabs.connect API (#22457)

* feat(extensions): add chrome.tabs.connect API

* test(extensions): verify that chrome.tabs.connect port communication works
This commit is contained in:
John Kleinschmidt 2020-03-05 09:59:32 -05:00 committed by GitHub
parent d6701ff435
commit 0201b3e571
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 62 additions and 0 deletions

View file

@ -138,6 +138,37 @@
} }
] ]
}, },
{
"name": "connect",
"nocompile": true,
"type": "function",
"description": "Connects to the content script(s) in the specified tab. The $(ref:runtime.onConnect) event is fired in each content script running in the specified tab for the current extension. For more details, see <a href='messaging'>Content Script Messaging</a>.",
"parameters": [
{
"type": "integer",
"name": "tabId",
"minimum": 0
},
{
"type": "object",
"name": "connectInfo",
"properties": {
"name": { "type": "string", "optional": true, "description": "Is passed into onConnect for content scripts that are listening for the connection event." },
"frameId": {
"type": "integer",
"optional": true,
"minimum": 0,
"description": "Open a port to a specific <a href='webNavigation#frame_ids'>frame</a> identified by <code>frameId</code> instead of all frames in the tab."
}
},
"optional": true
}
],
"returns": {
"$ref": "runtime.Port",
"description": "A port that can be used to communicate with the content scripts running in the specified tab. The port's $(ref:runtime.Port) event is fired if the tab closes or does not exist. "
}
},
{ {
"name": "executeScript", "name": "executeScript",
"type": "function", "type": "function",

View file

@ -186,6 +186,22 @@ ifdescribe(process.electronBinding('features').isExtensionsEnabled())('chrome ex
expect(response).to.equal(3) expect(response).to.equal(3)
}) })
it('connect', async () => {
const customSession = session.fromPartition(`persist:${require('uuid').v4()}`)
await customSession.loadExtension(path.join(fixtures, 'extensions', 'chrome-api'))
const w = new BrowserWindow({ show: false, webPreferences: { session: customSession, nodeIntegration: true } })
await w.loadURL(url)
const portName = require('uuid').v4()
const message = { method: 'connectTab', args: [portName] }
w.webContents.executeJavaScript(`window.postMessage('${JSON.stringify(message)}', '*')`)
const [,, responseString] = await emittedOnce(w.webContents, 'console-message')
const response = responseString.split(',')
expect(response[0]).to.equal(portName)
expect(response[1]).to.equal('howdy')
})
it('sendMessage receives the response', async function () { it('sendMessage receives the response', async function () {
const customSession = session.fromPartition(`persist:${require('uuid').v4()}`) const customSession = session.fromPartition(`persist:${require('uuid').v4()}`)
await customSession.loadExtension(path.join(fixtures, 'extensions', 'chrome-api')) await customSession.loadExtension(path.join(fixtures, 'extensions', 'chrome-api'))

View file

@ -16,6 +16,13 @@ chrome.runtime.onMessage.addListener((message, sender, sendResponse) => {
chrome.tabs.executeScript(tabId, { code }, ([result]) => sendResponse(result)) chrome.tabs.executeScript(tabId, { code }, ([result]) => sendResponse(result))
break break
} }
case 'connectTab': {
const [name] = args
const port = chrome.tabs.connect(tabId, { name })
port.postMessage('howdy')
break
}
} }
// Respond asynchronously // Respond asynchronously
return true return true

View file

@ -29,6 +29,14 @@ const testMap = {
chrome.runtime.sendMessage({ method: 'executeScript', args: [code] }, response => { chrome.runtime.sendMessage({ method: 'executeScript', args: [code] }, response => {
console.log(JSON.stringify(response)) console.log(JSON.stringify(response))
}) })
},
connectTab (name) {
chrome.runtime.onConnect.addListener(port => {
port.onMessage.addListener(message => {
console.log([port.name, message].join())
})
})
chrome.runtime.sendMessage({ method: 'connectTab', args: [name] })
} }
} }