From 90d62e5b98302915a0b70bd0f748684b4f108e5e Mon Sep 17 00:00:00 2001 From: Jeremy Apthorp Date: Wed, 28 Aug 2019 23:50:14 -0700 Subject: [PATCH] fix: nws13n: make ses.setUserAgent work (#20014) * refactor tests to better control window creation * fix: nws13n: make ses.setUserAgent work --- patches/chromium/.patches | 1 + ...xpose_setuseragent_on_networkcontext.patch | 90 ++++++++++++++++++ shell/browser/api/atom_api_session.cc | 6 +- spec-main/api-session-spec.ts | 91 +++++++++++++------ 4 files changed, 158 insertions(+), 30 deletions(-) create mode 100644 patches/chromium/expose_setuseragent_on_networkcontext.patch diff --git a/patches/chromium/.patches b/patches/chromium/.patches index ef462ef4252..4feb743aef9 100644 --- a/patches/chromium/.patches +++ b/patches/chromium/.patches @@ -76,3 +76,4 @@ add_contentgpuclient_precreatemessageloop_callback.patch picture-in-picture.patch disable_compositor_recycling.patch allow_new_privileges_in_unsandboxed_child_processes.patch +expose_setuseragent_on_networkcontext.patch diff --git a/patches/chromium/expose_setuseragent_on_networkcontext.patch b/patches/chromium/expose_setuseragent_on_networkcontext.patch new file mode 100644 index 00000000000..d220e96913c --- /dev/null +++ b/patches/chromium/expose_setuseragent_on_networkcontext.patch @@ -0,0 +1,90 @@ +From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001 +From: Jeremy Apthorp +Date: Wed, 28 Aug 2019 12:21:25 -0700 +Subject: expose SetUserAgent on NetworkContext + +Applying +https://chromium-review.googlesource.com/c/chromium/src/+/1585083 before +it's merged. There may end up being a better way of doing this, or the +patch may be merged upstream, at which point this patch should be +removed. + +diff --git a/net/url_request/static_http_user_agent_settings.h b/net/url_request/static_http_user_agent_settings.h +index 0ccfe130f00ec3b6c75cd8ee04d5a2777e1fd00c..653829457d58bf92057cc36aa8a289703d8b0577 100644 +--- a/net/url_request/static_http_user_agent_settings.h ++++ b/net/url_request/static_http_user_agent_settings.h +@@ -26,13 +26,17 @@ class NET_EXPORT StaticHttpUserAgentSettings : public HttpUserAgentSettings { + accept_language_ = new_accept_language; + } + ++ void set_user_agent(const std::string& new_user_agent) { ++ user_agent_ = new_user_agent; ++ } ++ + // HttpUserAgentSettings implementation + std::string GetAcceptLanguage() const override; + std::string GetUserAgent() const override; + + private: + std::string accept_language_; +- const std::string user_agent_; ++ std::string user_agent_; + + DISALLOW_COPY_AND_ASSIGN(StaticHttpUserAgentSettings); + }; +diff --git a/services/network/network_context.cc b/services/network/network_context.cc +index 20243f6c333478f14e5bff3789e780b996887512..34e4fd20f78ee0376053720742fc9af60dae37e3 100644 +--- a/services/network/network_context.cc ++++ b/services/network/network_context.cc +@@ -1095,6 +1095,13 @@ void NetworkContext::SetNetworkConditions( + std::move(network_conditions)); + } + ++void NetworkContext::SetUserAgent(const std::string& new_user_agent) { ++ // This may only be called on NetworkContexts created with a constructor that ++ // calls ApplyContextParamsToBuilder. ++ DCHECK(user_agent_settings_); ++ user_agent_settings_->set_user_agent(new_user_agent); ++} ++ + void NetworkContext::SetAcceptLanguage(const std::string& new_accept_language) { + // This may only be called on NetworkContexts created with the constructor + // that calls MakeURLRequestContext(). +diff --git a/services/network/network_context.h b/services/network/network_context.h +index f117ba5edfa7da25efd25e52ac5fff8b37ba3de3..5a805b4ee688ee7a113c833535db861b0e3b2ef9 100644 +--- a/services/network/network_context.h ++++ b/services/network/network_context.h +@@ -213,6 +213,7 @@ class COMPONENT_EXPORT(NETWORK_SERVICE) NetworkContext + void CloseIdleConnections(CloseIdleConnectionsCallback callback) override; + void SetNetworkConditions(const base::UnguessableToken& throttling_profile_id, + mojom::NetworkConditionsPtr conditions) override; ++ void SetUserAgent(const std::string& new_user_agent) override; + void SetAcceptLanguage(const std::string& new_accept_language) override; + void SetEnableReferrers(bool enable_referrers) override; + #if defined(OS_CHROMEOS) +diff --git a/services/network/public/mojom/network_context.mojom b/services/network/public/mojom/network_context.mojom +index 034b5720ffa497429b2cdfcc06cd2d421f2db99c..aa2c88443e744396e381377ec20b03fb0e66d2db 100644 +--- a/services/network/public/mojom/network_context.mojom ++++ b/services/network/public/mojom/network_context.mojom +@@ -857,6 +857,9 @@ interface NetworkContext { + SetNetworkConditions(mojo_base.mojom.UnguessableToken throttling_profile_id, + NetworkConditions? conditions); + ++ // Updates the user agent to be used for requests. ++ SetUserAgent(string new_user_agent); ++ + // Updates the Accept-Language header to be used for requests. + SetAcceptLanguage(string new_accept_language); + +diff --git a/services/network/test/test_network_context.h b/services/network/test/test_network_context.h +index aa21b8e434c01a866e3fae04b8de54a3cdfb725e..63880702102a396576c5825f8c5c7731cde091cc 100644 +--- a/services/network/test/test_network_context.h ++++ b/services/network/test/test_network_context.h +@@ -93,6 +93,7 @@ class TestNetworkContext : public mojom::NetworkContext { + void CloseIdleConnections(CloseIdleConnectionsCallback callback) override {} + void SetNetworkConditions(const base::UnguessableToken& throttling_profile_id, + mojom::NetworkConditionsPtr conditions) override {} ++ void SetUserAgent(const std::string& new_user_agent) override {} + void SetAcceptLanguage(const std::string& new_accept_language) override {} + void SetEnableReferrers(bool enable_referrers) override {} + #if defined(OS_CHROMEOS) diff --git a/shell/browser/api/atom_api_session.cc b/shell/browser/api/atom_api_session.cc index c367c85686f..6898dd33774 100644 --- a/shell/browser/api/atom_api_session.cc +++ b/shell/browser/api/atom_api_session.cc @@ -490,8 +490,10 @@ void Session::AllowNTLMCredentialsForDomains(const std::string& domains) { void Session::SetUserAgent(const std::string& user_agent, mate::Arguments* args) { - CHECK(false) - << "TODO: This was disabled when the network service was turned on"; + browser_context_->SetUserAgent(user_agent); + content::BrowserContext::GetDefaultStoragePartition(browser_context_.get()) + ->GetNetworkContext() + ->SetUserAgent(user_agent); } std::string Session::GetUserAgent() { diff --git a/spec-main/api-session-spec.ts b/spec-main/api-session-spec.ts index 47f08ec1600..95884c0468f 100644 --- a/spec-main/api-session-spec.ts +++ b/spec-main/api-session-spec.ts @@ -7,7 +7,7 @@ import * as ChildProcess from 'child_process' import { session, BrowserWindow, net, ipcMain, Session } from 'electron' import * as send from 'send' import * as auth from 'basic-auth' -import { closeWindow } from './window-helpers' +import { closeAllWindows } from './window-helpers' import { emittedOnce } from './events-helpers' import { AddressInfo } from 'net'; @@ -16,25 +16,8 @@ import { AddressInfo } from 'net'; describe('session module', () => { const fixtures = path.resolve(__dirname, '..', 'spec', 'fixtures') - let w: BrowserWindow const url = 'http://127.0.0.1' - beforeEach(() => { - w = new BrowserWindow({ - show: false, - width: 400, - height: 400, - webPreferences: { - nodeIntegration: true, - webviewTag: true, - } - }) - }) - - afterEach(() => { - return closeWindow(w).then(() => { w = null as any }) - }) - describe('session.defaultSession', () => { it('returns the default session', () => { expect(session.defaultSession).to.equal(session.fromPartition('')) @@ -73,6 +56,7 @@ describe('session module', () => { describe('ses.cookies', () => { const name = '0' const value = '0' + afterEach(closeAllWindows) it('should get cookies', async () => { const server = http.createServer((req, res) => { @@ -82,6 +66,7 @@ describe('session module', () => { }) await new Promise(resolve => server.listen(0, '127.0.0.1', resolve)) const { port } = server.address() as AddressInfo + const w = new BrowserWindow({ show: false }) await w.loadURL(`${url}:${port}`) const list = await w.webContents.session.cookies.get({ url }) const cookie = list.find(cookie => cookie.name === name) @@ -218,7 +203,9 @@ describe('session module', () => { }) describe('ses.clearStorageData(options)', () => { + afterEach(closeAllWindows) it('clears localstorage data', async () => { + const w = new BrowserWindow({ show: false, webPreferences: { nodeIntegration: true } }) await w.loadFile(path.join(fixtures, 'api', 'localstorage.html')) const options = { origin: 'file://', @@ -234,7 +221,9 @@ describe('session module', () => { }) describe('will-download event', () => { + afterEach(closeAllWindows) it('can cancel default download behavior', async () => { + const w = new BrowserWindow({ show: false }) const mockFile = Buffer.alloc(1024) const contentDisposition = 'inline; filename="mockFile.txt"' const downloadServer = http.createServer((req, res) => { @@ -276,14 +265,6 @@ describe('session module', () => { } beforeEach(async () => { - if (w != null) w.destroy() - w = new BrowserWindow({ - show: false, - webPreferences: { - partition: partitionName, - nodeIntegration: true, - } - }) customSession = session.fromPartition(partitionName) await customSession.protocol.registerStringProtocol(protocolName, handler) }) @@ -292,6 +273,7 @@ describe('session module', () => { await customSession.protocol.unregisterProtocol(protocolName) customSession = null as any }) + afterEach(closeAllWindows) it('does not affect defaultSession', async () => { const result1 = await protocol.isProtocolHandled(protocolName) @@ -302,6 +284,15 @@ describe('session module', () => { }) it('handles requests from partition', async () => { + const w = new BrowserWindow({ + show: false, + webPreferences: { + partition: partitionName, + nodeIntegration: true, + } + }) + customSession = session.fromPartition(partitionName) + await customSession.protocol.registerStringProtocol(protocolName, handler) w.loadURL(`${protocolName}://fake-host`) await emittedOnce(ipcMain, 'hello') }) @@ -384,6 +375,7 @@ describe('session module', () => { after(async () => { await protocol.unregisterProtocol(scheme) }) + afterEach(closeAllWindows) it('returns blob data for uuid', (done) => { const postData = JSON.stringify({ @@ -411,6 +403,7 @@ describe('session module', () => { } }, (error) => { if (error) return done(error) + const w = new BrowserWindow({ show: false }) w.loadURL(url) }) }) @@ -443,6 +436,7 @@ describe('session module', () => { session.defaultSession.setCertificateVerifyProc(null) server.close(done) }) + afterEach(closeAllWindows) it('accepts the request when the callback is called with 0', async () => { session.defaultSession.setCertificateVerifyProc(({ hostname, certificate, verificationResult, errorCode }, callback) => { @@ -451,6 +445,7 @@ describe('session module', () => { callback(0) }) + const w = new BrowserWindow({ show: false }) await w.loadURL(`https://127.0.0.1:${(server.address() as AddressInfo).port}`) expect(w.webContents.getTitle()).to.equal('hello') }) @@ -472,6 +467,7 @@ describe('session module', () => { }) const url = `https://127.0.0.1:${(server.address() as AddressInfo).port}` + const w = new BrowserWindow({ show: false }) await expect(w.loadURL(url)).to.eventually.be.rejectedWith(/ERR_FAILED/) expect(w.webContents.getTitle()).to.equal(url) }) @@ -484,6 +480,7 @@ describe('session module', () => { }) const url = `https://127.0.0.1:${(server.address() as AddressInfo).port}` + const w = new BrowserWindow({ show: false }) await expect(w.loadURL(url), 'first load').to.eventually.be.rejectedWith(/ERR_FAILED/) await emittedOnce(w.webContents, 'did-stop-loading') await expect(w.loadURL(url + '/test'), 'second load').to.eventually.be.rejectedWith(/ERR_FAILED/) @@ -555,6 +552,7 @@ describe('session module', () => { after(async () => { await new Promise(resolve => downloadServer.close(resolve)) }) + afterEach(closeAllWindows) const isPathEqual = (path1: string, path2: string) => { return path.relative(path1, path2) === '' @@ -591,6 +589,7 @@ describe('session module', () => { it('can download using WebContents.downloadURL', (done) => { const port = address.port + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.savePath = downloadFilePath item.on('done', function (e, state) { @@ -609,6 +608,7 @@ describe('session module', () => { } protocol.registerHttpProtocol(protocolName, handler, (error) => { if (error) return done(error) + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.savePath = downloadFilePath item.on('done', function (e, state) { @@ -622,6 +622,7 @@ describe('session module', () => { it('can download using WebView.downloadURL', async () => { const port = address.port + const w = new BrowserWindow({ show: false, webPreferences: { webviewTag: true } }) await w.loadURL('about:blank') function webviewDownload({fixtures, url, port}: {fixtures: string, url: string, port: string}) { const webview = new (window as any).WebView() @@ -646,6 +647,7 @@ describe('session module', () => { it('can cancel download', (done) => { const port = address.port + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.savePath = downloadFilePath item.on('done', function (e, state) { @@ -670,6 +672,7 @@ describe('session module', () => { } const port = address.port + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.savePath = downloadFilePath item.on('done', function (e, state) { @@ -700,6 +703,7 @@ describe('session module', () => { securityScopedBookmarks: true } + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.setSavePath(filePath) item.setSaveDialogOptions(options) @@ -714,6 +718,7 @@ describe('session module', () => { describe('when a save path is specified and the URL is unavailable', () => { it('does not display a save dialog and reports the done state as interrupted', (done) => { + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { item.savePath = downloadFilePath if (item.getState() === 'interrupted') { @@ -730,6 +735,7 @@ describe('session module', () => { }) describe('ses.createInterruptedDownload(options)', () => { + afterEach(closeAllWindows) it('can create an interrupted download item', (done) => { const downloadFilePath = path.join(__dirname, '..', 'fixtures', 'mock.pdf') const options = { @@ -739,6 +745,7 @@ describe('session module', () => { offset: 0, length: 5242880 } + const w = new BrowserWindow({ show: false }) w.webContents.session.once('will-download', function (e, item) { expect(item.getState()).to.equal('interrupted') item.cancel() @@ -762,6 +769,7 @@ describe('session module', () => { try { await new Promise(resolve => rangeServer.listen(0, '127.0.0.1', resolve)) const port = (rangeServer.address() as AddressInfo).port + const w = new BrowserWindow({ show: false }) const downloadCancelled: Promise = new Promise((resolve) => { w.webContents.session.once('will-download', function (e, item) { item.setSavePath(downloadFilePath) @@ -812,9 +820,9 @@ describe('session module', () => { }) describe('ses.setPermissionRequestHandler(handler)', () => { + afterEach(closeAllWindows) it('cancels any pending requests when cleared', async () => { - if (w != null) w.destroy() - w = new BrowserWindow({ + const w = new BrowserWindow({ show: false, webPreferences: { partition: `very-temp-permision-handler`, @@ -845,4 +853,31 @@ describe('session module', () => { expect(name).to.deep.equal('SecurityError') }) }) + + describe('ses.setUserAgent()', () => { + afterEach(closeAllWindows) + + it('can be retrieved with getUserAgent()', () => { + const userAgent = 'test-agent' + const ses = session.fromPartition(''+Math.random()) + ses.setUserAgent(userAgent) + expect(ses.getUserAgent()).to.equal(userAgent) + }) + + it('sets the User-Agent header for web requests made from renderers', async () => { + const userAgent = 'test-agent' + const ses = session.fromPartition(''+Math.random()) + ses.setUserAgent(userAgent) + const w = new BrowserWindow({ show: false, webPreferences: { session: ses } }) + let headers: http.IncomingHttpHeaders | null = null + const server = http.createServer((req, res) => { + headers = req.headers + res.end() + server.close() + }) + await new Promise(resolve => server.listen(0, '127.0.0.1', resolve)) + await w.loadURL(`http://127.0.0.1:${(server.address() as AddressInfo).port}`) + expect(headers!['user-agent']).to.equal(userAgent) + }) + }) })