netwerk/protocol/http/UserAgentOverrides.jsm

Wed, 31 Dec 2014 06:09:35 +0100

author
Michael Schloh von Bennewitz <michael@schloh.com>
date
Wed, 31 Dec 2014 06:09:35 +0100
changeset 0
6474c204b198
permissions
-rw-r--r--

Cloned upstream origin tor-browser at tor-browser-31.3.0esr-4.5-1-build1
revision ID fc1c9ff7c1b2defdbc039f12214767608f46423f for hacking purpose.

michael@0 1 /* This Source Code Form is subject to the terms of the Mozilla Public
michael@0 2 * License, v. 2.0. If a copy of the MPL was not distributed with this
michael@0 3 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
michael@0 4
michael@0 5 "use strict";
michael@0 6
michael@0 7 this.EXPORTED_SYMBOLS = [ "UserAgentOverrides" ];
michael@0 8
michael@0 9 const Ci = Components.interfaces;
michael@0 10 const Cc = Components.classes;
michael@0 11
michael@0 12 Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
michael@0 13 Components.utils.import("resource://gre/modules/Services.jsm");
michael@0 14 Components.utils.import("resource://gre/modules/UserAgentUpdates.jsm");
michael@0 15
michael@0 16 const OVERRIDE_MESSAGE = "Useragent:GetOverride";
michael@0 17 const PREF_OVERRIDES_ENABLED = "general.useragent.site_specific_overrides";
michael@0 18 const DEFAULT_UA = Cc["@mozilla.org/network/protocol;1?name=http"]
michael@0 19 .getService(Ci.nsIHttpProtocolHandler)
michael@0 20 .userAgent;
michael@0 21 const MAX_OVERRIDE_FOR_HOST_CACHE_SIZE = 250;
michael@0 22
michael@0 23 XPCOMUtils.defineLazyServiceGetter(this, "ppmm",
michael@0 24 "@mozilla.org/parentprocessmessagemanager;1",
michael@0 25 "nsIMessageListenerManager"); // Might have to make this broadcast?
michael@0 26
michael@0 27 var gPrefBranch;
michael@0 28 var gOverrides = new Map;
michael@0 29 var gUpdatedOverrides;
michael@0 30 var gOverrideForHostCache = new Map;
michael@0 31 var gInitialized = false;
michael@0 32 var gOverrideFunctions = [
michael@0 33 function (aHttpChannel) UserAgentOverrides.getOverrideForURI(aHttpChannel.URI)
michael@0 34 ];
michael@0 35 var gBuiltUAs = new Map;
michael@0 36
michael@0 37 this.UserAgentOverrides = {
michael@0 38 init: function uao_init() {
michael@0 39 if (gInitialized)
michael@0 40 return;
michael@0 41
michael@0 42 gPrefBranch = Services.prefs.getBranch("general.useragent.override.");
michael@0 43 gPrefBranch.addObserver("", buildOverrides, false);
michael@0 44
michael@0 45 ppmm.addMessageListener(OVERRIDE_MESSAGE, this);
michael@0 46 Services.prefs.addObserver(PREF_OVERRIDES_ENABLED, buildOverrides, false);
michael@0 47
michael@0 48 try {
michael@0 49 Services.obs.addObserver(HTTP_on_modify_request, "http-on-modify-request", false);
michael@0 50 } catch (x) {
michael@0 51 // The http-on-modify-request notification is disallowed in content processes.
michael@0 52 }
michael@0 53
michael@0 54 UserAgentUpdates.init(function(overrides) {
michael@0 55 gOverrideForHostCache.clear();
michael@0 56 if (overrides) {
michael@0 57 for (let domain in overrides) {
michael@0 58 overrides[domain] = getUserAgentFromOverride(overrides[domain]);
michael@0 59 }
michael@0 60 overrides.get = function(key) this[key];
michael@0 61 }
michael@0 62 gUpdatedOverrides = overrides;
michael@0 63 });
michael@0 64
michael@0 65 buildOverrides();
michael@0 66 gInitialized = true;
michael@0 67 },
michael@0 68
michael@0 69 addComplexOverride: function uao_addComplexOverride(callback) {
michael@0 70 // Add to front of array so complex overrides have precedence
michael@0 71 gOverrideFunctions.unshift(callback);
michael@0 72 },
michael@0 73
michael@0 74 getOverrideForURI: function uao_getOverrideForURI(aURI) {
michael@0 75 let host = aURI.asciiHost;
michael@0 76 if (!gInitialized ||
michael@0 77 (!gOverrides.size && !gUpdatedOverrides) ||
michael@0 78 !(host)) {
michael@0 79 return null;
michael@0 80 }
michael@0 81
michael@0 82 let override = gOverrideForHostCache.get(host);
michael@0 83 if (override !== undefined)
michael@0 84 return override;
michael@0 85
michael@0 86 function findOverride(overrides) {
michael@0 87 let searchHost = host;
michael@0 88 let userAgent = overrides.get(searchHost);
michael@0 89
michael@0 90 while (!userAgent) {
michael@0 91 let dot = searchHost.indexOf('.');
michael@0 92 if (dot === -1) {
michael@0 93 return null;
michael@0 94 }
michael@0 95 searchHost = searchHost.slice(dot + 1);
michael@0 96 userAgent = overrides.get(searchHost);
michael@0 97 }
michael@0 98 return userAgent;
michael@0 99 }
michael@0 100
michael@0 101 override = (gOverrides.size && findOverride(gOverrides))
michael@0 102 || (gUpdatedOverrides && findOverride(gUpdatedOverrides));
michael@0 103
michael@0 104 if (gOverrideForHostCache.size >= MAX_OVERRIDE_FOR_HOST_CACHE_SIZE) {
michael@0 105 gOverrideForHostCache.clear();
michael@0 106 }
michael@0 107 gOverrideForHostCache.set(host, override);
michael@0 108
michael@0 109 return override;
michael@0 110 },
michael@0 111
michael@0 112 uninit: function uao_uninit() {
michael@0 113 if (!gInitialized)
michael@0 114 return;
michael@0 115 gInitialized = false;
michael@0 116
michael@0 117 gPrefBranch.removeObserver("", buildOverrides);
michael@0 118
michael@0 119 Services.prefs.removeObserver(PREF_OVERRIDES_ENABLED, buildOverrides);
michael@0 120
michael@0 121 Services.obs.removeObserver(HTTP_on_modify_request, "http-on-modify-request");
michael@0 122 },
michael@0 123
michael@0 124 receiveMessage: function(aMessage) {
michael@0 125 let name = aMessage.name;
michael@0 126 switch (name) {
michael@0 127 case OVERRIDE_MESSAGE:
michael@0 128 let uri = aMessage.data.uri;
michael@0 129 return this.getOverrideForURI(uri);
michael@0 130 default:
michael@0 131 throw("Wrong Message in UserAgentOverride: " + name);
michael@0 132 }
michael@0 133 }
michael@0 134 };
michael@0 135
michael@0 136 function getUserAgentFromOverride(override)
michael@0 137 {
michael@0 138 let userAgent = gBuiltUAs.get(override);
michael@0 139 if (userAgent !== undefined) {
michael@0 140 return userAgent;
michael@0 141 }
michael@0 142 let [search, replace] = override.split("#", 2);
michael@0 143 if (search && replace) {
michael@0 144 userAgent = DEFAULT_UA.replace(new RegExp(search, "g"), replace);
michael@0 145 } else {
michael@0 146 userAgent = override;
michael@0 147 }
michael@0 148 gBuiltUAs.set(override, userAgent);
michael@0 149 return userAgent;
michael@0 150 }
michael@0 151
michael@0 152 function buildOverrides() {
michael@0 153 gOverrides.clear();
michael@0 154 gOverrideForHostCache.clear();
michael@0 155
michael@0 156 if (!Services.prefs.getBoolPref(PREF_OVERRIDES_ENABLED))
michael@0 157 return;
michael@0 158
michael@0 159 let builtUAs = new Map;
michael@0 160 let domains = gPrefBranch.getChildList("");
michael@0 161
michael@0 162 for (let domain of domains) {
michael@0 163 let override = gPrefBranch.getCharPref(domain);
michael@0 164 let userAgent = getUserAgentFromOverride(override);
michael@0 165
michael@0 166 if (userAgent != DEFAULT_UA) {
michael@0 167 gOverrides.set(domain, userAgent);
michael@0 168 }
michael@0 169 }
michael@0 170 }
michael@0 171
michael@0 172 function HTTP_on_modify_request(aSubject, aTopic, aData) {
michael@0 173 let channel = aSubject.QueryInterface(Ci.nsIHttpChannel);
michael@0 174
michael@0 175 for (let callback of gOverrideFunctions) {
michael@0 176 let modifiedUA = callback(channel, DEFAULT_UA);
michael@0 177 if (modifiedUA) {
michael@0 178 channel.setRequestHeader("User-Agent", modifiedUA, false);
michael@0 179 return;
michael@0 180 }
michael@0 181 }
michael@0 182 }

mercurial