使用Chrome扩展程序更改navigator.userAgent

时间:2014-04-21 17:13:06

标签: javascript html google-chrome google-chrome-extension

我正在尝试使用简单的chrome扩展名重载navigator.userAgent。由于内容脚本在隔离的env中运行,我尝试创建一个脚本元素并将逻辑写入此脚本。这发生在扩展程序的后台页面

chrome.tabs.query({
  active:!0
}, function(tabs) {
    var x = "window.navigator.__defineGetter__('userAgent', function() {" +
            "return 'Mozilla/5.0 (Linux; Android 4.2.1; en-us; Nexus 5 Build/JOP40D)" +
            " AppleWebKit/535.19 (KHTML, like Gecko) Chrome/18.0.1025.166 Mobile " + 
            "Safari/535.19'; });console.log(navigator.userAgent);";

    for (var i = 0;i < tabs.length;i++) {
      var code = 'var s = document.createElement("script"); s.text = "' + x +
                 '"; document.head.insertBefore(s, document.head.firstChild);' + 
                 'navigator.userAgent ="s"; console.log(navigator.userAgent);';

      // Inject into the tabs of choice - currently everything.
      chrome.tabs.executeScript(tabs[i].id, {
        code: code
      });
    }
  });

脚本附加了head元素,我可以看到UA字符串是通过在chrome的控制台中尝试navigator.userAgent而被欺骗的字符串,因此我相信导航器对象已经过载。

但这似乎不是有效的方式或根本没有发生,因为导航器对象没有更新,我发现通过 - http://www.quirksmode.org/js/detect.html仍然显示了Mac的UA。

那么,我到底错过了什么?

2 个答案:

答案 0 :(得分:11)

navigator.userAgent是只读属性。如果要更改navigator.userAgent,则需要创建新对象并复制属性,或创建新对象并从navigator继承并分配新的getter / setter。

我最近创建了这样的扩展程序。我在Linux上,但我偶尔会下载Chrome for Windows。以下扩展程序会在Chrome的下载页面上将用户代理更改为Windows XP:

contentscript.js

var actualCode =  '(' + function() {
    'use strict';
    var navigator = window.navigator;
    var modifiedNavigator;
    if ('userAgent' in Navigator.prototype) {
        // Chrome 43+ moved all properties from navigator to the prototype,
        // so we have to modify the prototype instead of navigator.
        modifiedNavigator = Navigator.prototype;

    } else {
        // Chrome 42- defined the property on navigator.
        modifiedNavigator = Object.create(navigator);
        Object.defineProperty(window, 'navigator', {
            value: modifiedNavigator,
            configurable: false,
            enumerable: false,
            writable: false
        });
    }
    // Pretend to be Windows XP
    Object.defineProperties(modifiedNavigator, {
        userAgent: {
            value: navigator.userAgent.replace(/\([^)]+\)/, 'Windows NT 5.1'),
            configurable: false,
            enumerable: true,
            writable: false
        },
        appVersion: {
            value: navigator.appVersion.replace(/\([^)]+\)/, 'Windows NT 5.1'),
            configurable: false,
            enumerable: true,
            writable: false
        },
        platform: {
            value: 'Win32',
            configurable: false,
            enumerable: true,
            writable: false
        },
    });
} + ')();';

var s = document.createElement('script');
s.textContent = actualCode;
document.documentElement.appendChild(s);
s.remove();

的manifest.json

{
    "name": "navigator.userAgent",
    "description": "Change navigator.userAgent to Windows on Chrome's download page.",
    "version": "1",
    "manifest_version": 2,
    "content_scripts": [{
        "run_at": "document_start",
        "js": ["contentscript.js"],
        "matches": [
            "*://www.google.com/intl/*/chrome/browser/*"
        ]
    }]
}

正如您所看到的,我是declaring the content script而不是动态插入它,以确保我的代码在页面加载之前运行。此外,我正在使用其中一个技巧described in this answer来更改页面的navigator,而不是在隔离的内容脚本世界中更改其他navigator

请注意,这只会修改从JavaScript看到的userAgent。如果要修改发送到服务器的用户代理,请查看Associate a custom user agent to a specific Google Chrome page/tab

答案 1 :(得分:0)

您是否尝试更改在请求中发送的User-Agent标头?您必须使用declarativeWebRequest或webRequest API。

仅在发送请求后才会在页面中运行内容脚本。