diff options
Diffstat (limited to 'netwerk/protocol/http/UserAgentOverrides.jsm')
-rw-r--r-- | netwerk/protocol/http/UserAgentOverrides.jsm | 182 |
1 files changed, 182 insertions, 0 deletions
diff --git a/netwerk/protocol/http/UserAgentOverrides.jsm b/netwerk/protocol/http/UserAgentOverrides.jsm new file mode 100644 index 0000000000..22c676f068 --- /dev/null +++ b/netwerk/protocol/http/UserAgentOverrides.jsm @@ -0,0 +1,182 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +"use strict"; + +this.EXPORTED_SYMBOLS = [ "UserAgentOverrides" ]; + +const Ci = Components.interfaces; +const Cc = Components.classes; + +Components.utils.import("resource://gre/modules/XPCOMUtils.jsm"); +Components.utils.import("resource://gre/modules/Services.jsm"); +Components.utils.import("resource://gre/modules/UserAgentUpdates.jsm"); + +const OVERRIDE_MESSAGE = "Useragent:GetOverride"; +const PREF_OVERRIDES_ENABLED = "general.useragent.site_specific_overrides"; +const DEFAULT_UA = Cc["@mozilla.org/network/protocol;1?name=http"] + .getService(Ci.nsIHttpProtocolHandler) + .userAgent; +const MAX_OVERRIDE_FOR_HOST_CACHE_SIZE = 250; + +XPCOMUtils.defineLazyServiceGetter(this, "ppmm", + "@mozilla.org/parentprocessmessagemanager;1", + "nsIMessageListenerManager"); // Might have to make this broadcast? + +var gPrefBranch; +var gOverrides = new Map; +var gUpdatedOverrides; +var gOverrideForHostCache = new Map; +var gInitialized = false; +var gOverrideFunctions = [ + function (aHttpChannel) { return UserAgentOverrides.getOverrideForURI(aHttpChannel.URI); } +]; +var gBuiltUAs = new Map; + +this.UserAgentOverrides = { + init: function uao_init() { + if (gInitialized) + return; + + gPrefBranch = Services.prefs.getBranch("general.useragent.override."); + gPrefBranch.addObserver("", buildOverrides, false); + + ppmm.addMessageListener(OVERRIDE_MESSAGE, this); + Services.prefs.addObserver(PREF_OVERRIDES_ENABLED, buildOverrides, false); + + try { + Services.obs.addObserver(HTTP_on_useragent_request, "http-on-useragent-request", false); + } catch (x) { + // The http-on-useragent-request notification is disallowed in content processes. + } + + UserAgentUpdates.init(function(overrides) { + gOverrideForHostCache.clear(); + if (overrides) { + for (let domain in overrides) { + overrides[domain] = getUserAgentFromOverride(overrides[domain]); + } + overrides.get = function(key) { return this[key]; }; + } + gUpdatedOverrides = overrides; + }); + + buildOverrides(); + gInitialized = true; + }, + + addComplexOverride: function uao_addComplexOverride(callback) { + // Add to front of array so complex overrides have precedence + gOverrideFunctions.unshift(callback); + }, + + getOverrideForURI: function uao_getOverrideForURI(aURI) { + let host = aURI.asciiHost; + if (!gInitialized || + (!gOverrides.size && !gUpdatedOverrides) || + !(host)) { + return null; + } + + let override = gOverrideForHostCache.get(host); + if (override !== undefined) + return override; + + function findOverride(overrides) { + let searchHost = host; + let userAgent = overrides.get(searchHost); + + while (!userAgent) { + let dot = searchHost.indexOf('.'); + if (dot === -1) { + return null; + } + searchHost = searchHost.slice(dot + 1); + userAgent = overrides.get(searchHost); + } + return userAgent; + } + + override = (gOverrides.size && findOverride(gOverrides)) + || (gUpdatedOverrides && findOverride(gUpdatedOverrides)); + + if (gOverrideForHostCache.size >= MAX_OVERRIDE_FOR_HOST_CACHE_SIZE) { + gOverrideForHostCache.clear(); + } + gOverrideForHostCache.set(host, override); + + return override; + }, + + uninit: function uao_uninit() { + if (!gInitialized) + return; + gInitialized = false; + + gPrefBranch.removeObserver("", buildOverrides); + + Services.prefs.removeObserver(PREF_OVERRIDES_ENABLED, buildOverrides); + + Services.obs.removeObserver(HTTP_on_useragent_request, "http-on-useragent-request"); + }, + + receiveMessage: function(aMessage) { + let name = aMessage.name; + switch (name) { + case OVERRIDE_MESSAGE: + let uri = Services.io.newURI(aMessage.data.uri, null, null); + return this.getOverrideForURI(uri); + default: + throw("Wrong Message in UserAgentOverride: " + name); + } + } +}; + +function getUserAgentFromOverride(override) +{ + let userAgent = gBuiltUAs.get(override); + if (userAgent !== undefined) { + return userAgent; + } + let [search, replace] = override.split("#", 2); + if (search && replace) { + userAgent = DEFAULT_UA.replace(new RegExp(search, "g"), replace); + } else { + userAgent = override; + } + gBuiltUAs.set(override, userAgent); + return userAgent; +} + +function buildOverrides() { + gOverrides.clear(); + gOverrideForHostCache.clear(); + + if (!Services.prefs.getBoolPref(PREF_OVERRIDES_ENABLED)) + return; + + let builtUAs = new Map; + let domains = gPrefBranch.getChildList(""); + + for (let domain of domains) { + let override = gPrefBranch.getCharPref(domain); + let userAgent = getUserAgentFromOverride(override); + + if (userAgent != DEFAULT_UA) { + gOverrides.set(domain, userAgent); + } + } +} + +function HTTP_on_useragent_request(aSubject, aTopic, aData) { + let channel = aSubject.QueryInterface(Ci.nsIHttpChannel); + + for (let callback of gOverrideFunctions) { + let modifiedUA = callback(channel, DEFAULT_UA); + if (modifiedUA) { + channel.setRequestHeader("User-Agent", modifiedUA, false); + return; + } + } +} |