mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
a25085b248
- Bug 1195173 - Test asyncopen2 security checks for stylesheets (r=bz,ehsan) (358ae850a4) - Bug 1223644 - Clean up the nsSVGClipPathFrame reference loop detection code. r=longsonr (65042c3148) - Bug 1157064 - font-display descriptor parsing. r=dbaron (18f63d9244) - Bug 1242523 - Guard against GetWidget getting called with a null shell. r=snorp (55de891c6c) - Bug 1247679, part 3 - Replace NS_IMPL_CYCLE_COLLECTION_TRACE_JSVAL_MEMBER_CALLBACK with JS_MEMBER. r=smaug (09435e582c) - Bug 1247515 - Check nsScriptErrorWithStack's mStack member for null before tracing; r=mccr8 (0cb1e09aa9) - Bug 1178803. Fix the handling of the 'length' key in IDB keypaths when operating on a string. r=bent (69f15272a8) - Bug 1240660 - Remove "+" prefixes in jar manifests. r=gps (27de5a6a87) - Bug 1240660 - Remove support for the "+" prefix in jar manifests. r=gps (2bdf115c40) - 1240660 fallout fix (3b1c1e2d8b) - Back out the videocontrols piece of bug 1231828 due to b2g breakage and add to the ignore list. r=backout (bffe92e32c) - Bug 1201037 - coalesce network-events on Windows, r=mcmanus (5f48aab5c3) - Bug 1131626, fix autoscroll tests to work in e10s, r=felipe (c47adbaa10) - Bug 1231529 - Increase the timeout of browser_bug295977_autoscroll_overflow.js (691d27224d) - Bug 416611 - Changed BookmarkHTMLUtils.jsm to import bookmark tags from HTML document. r=mak (49a0accc13) - Bug 1130858 - Recipient autocomplete suggestion overrides ANY manual address input if quickly entered/pasted and confirmed with Enter/Tab before autocomplete suggestions disappear. r=mak (308196e116) - Bug 1197361. Optimize page thumbnails based on screen size. r=ttaubert (29dca20366) - Bug 555087 - Add check for exception when passing undefined parameter. r=adw (3588477c56) - Bug 750544 - Remove the scope bar from the Library window. r=mano. (e1544ca40a) - Bug 1175678 - Update icons for different Mixed Content Blocking states in the URL bar r=MattN (79a8c1d70a) - remove parent folder (95af2afea6) - Bug 1145063 - Remove the keywords column from the Library r=mak (600f18004a) - Bug 1203803: Force cleanup for specific states only; r=khuey a=CLOSED TREE (2f9f78ad72) - Bug 1203803 - Remove forced cleanup from FactoryOp::ActorDestroy() since it cause more harm than good; r=khuey (76a00e58cb) - Bug 1195149 - Upgrade the check to a MOZ_RELEASE_ASSERT. r=janv (071d1fc267) - Bug 1185223 - crash at [@ mozilla::dom::quota::QuotaObject::Release() ]; r=khuey (d460972a45) - Bug 1229376 - Downgrade lastVacuumSize > 0 assertion to an NS_ASSERTION so we don't have to star it all over the place, rs=khuey (81d715ab71) - Bug 1239666 - part 1, get rid of the default parameter. r=waldo (639fb79ec3) - Bug 1239666 - part 2, dom/indexedDB change. r=khuey (6faaf25df4) - Bug 1239666 - part 3, devtools test case changes. r=sphink (c010d06a77) - Bug 1248309 - Fix caret size not updated when only zoom level is changed. r=roc (91cc5b35df) - bits of Bug 983623 - Implement cmd_un/redo for the new async TM (7fd41a9c37) - Bug 1245649: Turn on no-trailing-spaces. r=Gijs (7f87c967af) - Bug 1245649: Enable no-negated-in-lhs, no-native-reassign, no-func-assign and no-labels. r=MattN (5f801e4a4c) - Bug 1177639 - ensure copy is made visible again, r=mak (a0a3238cc1) - Bug 993274 - Remove cmd_new:livemark. r=mak (2a29f447d5) - more openparent removal (e8f0c74bbd) - minor fixes (b80d0307e9) - hide pocket, we don't support it anyway (8a802e9e5b) - Bug 1197966 - Fix typo when releasing content-side probes in PerformanceStats-content.js. r=felipe (9241324efd) - Bug 1219144 - Performance alerts are now labelled with isJankVisible; r=avih,froydnj (c1549a24f5) - Bug 1219144 - Using the nsRefreshDriver's jank indication for performance monitoring;f?froydnj r=froydnj (735c6fba9c) - Bug 1211783 - Add KeyframeEffect interface to dom/webidl/KeyframeEffect.webidl. r=smaug,birtles (fca332fea0) - Fix up missing dependency in bug 1247162. r=me (897f1ebd9b) - Bug 795681 - Print out failures in mozunit runner. r=gps (ce418e5ea8) - Bug 801679 - Handle expected failures and skipped tests in mozunit runner. r=gps (396ca02893) - Bug 1247833 - Display the class name in mozunit output. r=gps (0b5724f41c) - Bug 1245022 - Kill stlport's Makefile.in. r=mshal (225f662efc) - Bug 1194603 - Remove INTERNAL_TOOLS; r=mshal (e8e90ec1c3) - Bug 1247994 - Upgrade vendored requests package to 2.9.1; r=mshal (591a9e849a) - Bug 1234612: Print path when failing to create virtualenv r=gps (ae69d82c1f) - Bug 1235109 - Remove support for -I in preprocessor and jar maker. r=gps (a6c60bb407) - Bug 1248027 - '#define FOO' should use an empty value, not '1'; r=glandium (8a3a33bad9) - Bug 1239872 - Prevent jar maker from installing the same file twice. r=gps (4c3e496212) - Bug 1224450 - Skip difficult directories in the CompileDB. r=gps (4fe09cff61) - Bug 1245015 - Properly handle ObjC sources in the CompileDB backend. r=mshal (9d2f312c01) - Bug 1224450 - Ignore host compilations in the CompileDB backend. r=gps (c0e6d114b5) - Bug 1224450 - Make the CompileDB derive its commands from the moz.build data. r=gps (abee41178f) - Bug 1247743 - Expose non-pinning JS_Atomize[UC]String JSAPI functions; r=terrence (66aa23066d) - Bug 1230071 - Enable warnings-as-errors in js/src. r=Waldo (a0c8acf6ad) - Bug 1007136 - Ensure malloc/free always match when using JSAutoByteString; r=bz (81dfcf036a) - Bug 1246850 - check the NotifyIpInterfaceChange() return code, r=mcmanus (bc224f287c) - Bug 739029 - null check a thread allocation in notifyaddrlistener r=bagder (ce0ddfc44c)
414 lines
13 KiB
JavaScript
414 lines
13 KiB
JavaScript
// -*- Mode: js; tab-width: 2; indent-tabs-mode: nil; js2-basic-offset: 2; js2-skip-preprocessor-directives: t; -*-
|
|
/* 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 = ["SimpleServiceDiscovery"];
|
|
|
|
const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/Timer.jsm");
|
|
|
|
let log = Cu.reportError;
|
|
|
|
XPCOMUtils.defineLazyGetter(this, "converter", function () {
|
|
let conv = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter);
|
|
conv.charset = "utf8";
|
|
return conv;
|
|
});
|
|
|
|
// Spec information:
|
|
// https://tools.ietf.org/html/draft-cai-ssdp-v1-03
|
|
// http://www.dial-multiscreen.org/dial-protocol-specification
|
|
const SSDP_PORT = 1900;
|
|
const SSDP_ADDRESS = "239.255.255.250";
|
|
|
|
const SSDP_DISCOVER_PACKET =
|
|
"M-SEARCH * HTTP/1.1\r\n" +
|
|
"HOST: " + SSDP_ADDRESS + ":" + SSDP_PORT + "\r\n" +
|
|
"MAN: \"ssdp:discover\"\r\n" +
|
|
"MX: 2\r\n" +
|
|
"ST: %SEARCH_TARGET%\r\n\r\n";
|
|
|
|
const SSDP_DISCOVER_ATTEMPTS = 3;
|
|
const SSDP_DISCOVER_DELAY = 500;
|
|
const SSDP_DISCOVER_TIMEOUT_MULTIPLIER = 2;
|
|
const SSDP_TRANSMISSION_INTERVAL = 1000;
|
|
|
|
const EVENT_SERVICE_FOUND = "ssdp-service-found";
|
|
const EVENT_SERVICE_LOST = "ssdp-service-lost";
|
|
|
|
/*
|
|
* SimpleServiceDiscovery manages any discovered SSDP services. It uses a UDP
|
|
* broadcast to locate available services on the local network.
|
|
*/
|
|
var SimpleServiceDiscovery = {
|
|
get EVENT_SERVICE_FOUND() { return EVENT_SERVICE_FOUND; },
|
|
get EVENT_SERVICE_LOST() { return EVENT_SERVICE_LOST; },
|
|
|
|
_devices: new Map(),
|
|
_services: new Map(),
|
|
_searchSocket: null,
|
|
_searchInterval: 0,
|
|
_searchTimestamp: 0,
|
|
_searchTimeout: Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer),
|
|
_searchRepeat: Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer),
|
|
|
|
_forceTrailingSlash: function(aURL) {
|
|
// Cleanup the URL to make it consistent across devices
|
|
try {
|
|
aURL = Services.io.newURI(aURL, null, null).spec;
|
|
} catch(e) {}
|
|
return aURL;
|
|
},
|
|
|
|
// nsIUDPSocketListener implementation
|
|
onPacketReceived: function(aSocket, aMessage) {
|
|
// Listen for responses from specific devices. There could be more than one
|
|
// available.
|
|
let response = aMessage.data.split("\n");
|
|
let service = {};
|
|
response.forEach(function(row) {
|
|
let name = row.toUpperCase();
|
|
if (name.startsWith("LOCATION")) {
|
|
service.location = row.substr(10).trim();
|
|
} else if (name.startsWith("ST")) {
|
|
service.target = row.substr(4).trim();
|
|
}
|
|
}.bind(this));
|
|
|
|
if (service.location && service.target) {
|
|
service.location = this._forceTrailingSlash(service.location);
|
|
|
|
// When we find a valid response, package up the service information
|
|
// and pass it on.
|
|
try {
|
|
this._processService(service);
|
|
} catch (e) {}
|
|
}
|
|
},
|
|
|
|
onStopListening: function(aSocket, aStatus) {
|
|
// This is fired when the socket is closed expectedly or unexpectedly.
|
|
// nsITimer.cancel() is a no-op if the timer is not active.
|
|
this._searchTimeout.cancel();
|
|
this._searchSocket = null;
|
|
},
|
|
|
|
// Start a search. Make it continuous by passing an interval (in milliseconds).
|
|
// This will stop a current search loop because the timer resets itself.
|
|
// Returns the existing search interval.
|
|
search: function search(aInterval) {
|
|
let existingSearchInterval = this._searchInterval;
|
|
if (aInterval > 0) {
|
|
this._searchInterval = aInterval || 0;
|
|
this._searchRepeat.initWithCallback(this._search.bind(this), this._searchInterval, Ci.nsITimer.TYPE_REPEATING_SLACK);
|
|
}
|
|
this._search();
|
|
return existingSearchInterval;
|
|
},
|
|
|
|
// Stop the current continuous search
|
|
stopSearch: function stopSearch() {
|
|
this._searchRepeat.cancel();
|
|
},
|
|
|
|
_usingLAN: function() {
|
|
let network = Cc["@mozilla.org/network/network-link-service;1"].getService(Ci.nsINetworkLinkService);
|
|
return (network.linkType == Ci.nsINetworkLinkService.LINK_TYPE_WIFI ||
|
|
network.linkType == Ci.nsINetworkLinkService.LINK_TYPE_ETHERNET ||
|
|
network.linkType == Ci.nsINetworkLinkService.LINK_TYPE_UNKNOWN);
|
|
},
|
|
|
|
_search: function _search() {
|
|
// If a search is already active, shut it down.
|
|
this._searchShutdown();
|
|
|
|
// We only search if on local network
|
|
if (!this._usingLAN()) {
|
|
return;
|
|
}
|
|
|
|
// Update the timestamp so we can use it to clean out stale services the
|
|
// next time we search.
|
|
this._searchTimestamp = Date.now();
|
|
|
|
// Look for any fixed IP devices. Some routers might be configured to block
|
|
// UDP broadcasts, so this is a way to skip discovery.
|
|
this._searchFixedDevices();
|
|
|
|
// Perform a UDP broadcast to search for SSDP devices
|
|
let socket = Cc["@mozilla.org/network/udp-socket;1"].createInstance(Ci.nsIUDPSocket);
|
|
try {
|
|
socket.init(SSDP_PORT, false, Services.scriptSecurityManager.getSystemPrincipal());
|
|
socket.joinMulticast(SSDP_ADDRESS);
|
|
socket.asyncListen(this);
|
|
} catch (e) {
|
|
// We were unable to create the broadcast socket. Just return, but don't
|
|
// kill the interval timer. This might work next time.
|
|
log("failed to start socket: " + e);
|
|
return;
|
|
}
|
|
|
|
// Make the timeout SSDP_DISCOVER_TIMEOUT_MULTIPLIER times as long as the time needed to send out the discovery packets.
|
|
const SSDP_DISCOVER_TIMEOUT = this._devices.size * SSDP_DISCOVER_ATTEMPTS * SSDP_TRANSMISSION_INTERVAL * SSDP_DISCOVER_TIMEOUT_MULTIPLIER;
|
|
this._searchSocket = socket;
|
|
this._searchTimeout.initWithCallback(this._searchShutdown.bind(this), SSDP_DISCOVER_TIMEOUT, Ci.nsITimer.TYPE_ONE_SHOT);
|
|
|
|
let data = SSDP_DISCOVER_PACKET;
|
|
|
|
// Send discovery packets out at 1 per SSDP_TRANSMISSION_INTERVAL and send each SSDP_DISCOVER_ATTEMPTS times
|
|
// to allow for packet loss on noisy networks.
|
|
let timeout = SSDP_DISCOVER_DELAY;
|
|
for (let attempts = 0; attempts < SSDP_DISCOVER_ATTEMPTS; attempts++) {
|
|
for (let [key, device] of this._devices) {
|
|
let target = device.target;
|
|
setTimeout(function() {
|
|
let msgData = data.replace("%SEARCH_TARGET%", target);
|
|
try {
|
|
let msgRaw = converter.convertToByteArray(msgData);
|
|
socket.send(SSDP_ADDRESS, SSDP_PORT, msgRaw, msgRaw.length);
|
|
} catch (e) {
|
|
log("failed to convert to byte array: " + e);
|
|
}
|
|
}, timeout);
|
|
timeout += SSDP_TRANSMISSION_INTERVAL;
|
|
}
|
|
}
|
|
},
|
|
|
|
_searchFixedDevices: function _searchFixedDevices() {
|
|
let fixedDevices = null;
|
|
try {
|
|
fixedDevices = Services.prefs.getCharPref("browser.casting.fixedDevices");
|
|
} catch (e) {}
|
|
|
|
if (!fixedDevices) {
|
|
return;
|
|
}
|
|
|
|
fixedDevices = JSON.parse(fixedDevices);
|
|
for (let fixedDevice of fixedDevices) {
|
|
// Verify we have the right data
|
|
if (!("location" in fixedDevice) || !("target" in fixedDevice)) {
|
|
continue;
|
|
}
|
|
|
|
fixedDevice.location = this._forceTrailingSlash(fixedDevice.location);
|
|
|
|
let service = {
|
|
location: fixedDevice.location,
|
|
target: fixedDevice.target
|
|
};
|
|
|
|
// We don't assume the fixed target is ready. We still need to ping it.
|
|
try {
|
|
this._processService(service);
|
|
} catch (e) {}
|
|
}
|
|
},
|
|
|
|
// Called when the search timeout is hit. We use it to cleanup the socket and
|
|
// perform some post-processing on the services list.
|
|
_searchShutdown: function _searchShutdown() {
|
|
if (this._searchSocket) {
|
|
// This will call onStopListening.
|
|
this._searchSocket.close();
|
|
|
|
// Clean out any stale services
|
|
for (let [key, service] of this._services) {
|
|
if (service.lastPing != this._searchTimestamp) {
|
|
this.removeService(service.uuid);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
getSupportedExtensions: function() {
|
|
let extensions = [];
|
|
this.services.forEach(function(service) {
|
|
extensions = extensions.concat(service.extensions);
|
|
}, this);
|
|
return extensions.filter(function(extension, pos) {
|
|
return extensions.indexOf(extension) == pos;
|
|
});
|
|
},
|
|
|
|
getSupportedMimeTypes: function() {
|
|
let types = [];
|
|
this.services.forEach(function(service) {
|
|
types = types.concat(service.types);
|
|
}, this);
|
|
return types.filter(function(type, pos) {
|
|
return types.indexOf(type) == pos;
|
|
});
|
|
},
|
|
|
|
registerDevice: function registerDevice(aDevice) {
|
|
// We must have "id", "target" and "factory" defined
|
|
if (!("id" in aDevice) || !("target" in aDevice) || !("factory" in aDevice)) {
|
|
// Fatal for registration
|
|
throw "Registration requires an id, a target and a location";
|
|
}
|
|
|
|
// Only add if we don't already know about this device
|
|
if (!this._devices.has(aDevice.id)) {
|
|
this._devices.set(aDevice.id, aDevice);
|
|
} else {
|
|
log("device was already registered: " + aDevice.id);
|
|
}
|
|
},
|
|
|
|
unregisterDevice: function unregisterDevice(aDevice) {
|
|
// We must have "id", "target" and "factory" defined
|
|
if (!("id" in aDevice) || !("target" in aDevice) || !("factory" in aDevice)) {
|
|
return;
|
|
}
|
|
|
|
// Only remove if we know about this device
|
|
if (this._devices.has(aDevice.id)) {
|
|
this._devices.delete(aDevice.id);
|
|
} else {
|
|
log("device was not registered: " + aDevice.id);
|
|
}
|
|
},
|
|
|
|
findAppForService: function findAppForService(aService) {
|
|
if (!aService || !aService.deviceID) {
|
|
return null;
|
|
}
|
|
|
|
// Find the registration for the device
|
|
if (this._devices.has(aService.deviceID)) {
|
|
return this._devices.get(aService.deviceID).factory(aService);
|
|
}
|
|
return null;
|
|
},
|
|
|
|
findServiceForID: function findServiceForID(aUUID) {
|
|
if (this._services.has(aUUID)) {
|
|
return this._services.get(aUUID);
|
|
}
|
|
return null;
|
|
},
|
|
|
|
// Returns an array copy of the active services
|
|
get services() {
|
|
let array = [];
|
|
for (let [key, service] of this._services) {
|
|
let target = this._devices.get(service.deviceID);
|
|
service.extensions = target.extensions;
|
|
service.types = target.types;
|
|
array.push(service);
|
|
}
|
|
return array;
|
|
},
|
|
|
|
// Returns false if the service does not match the device's filters
|
|
_filterService: function _filterService(aService) {
|
|
// Loop over all the devices, looking for one that matches the service
|
|
for (let [key, device] of this._devices) {
|
|
// First level of match is on the target itself
|
|
if (device.target != aService.target) {
|
|
continue;
|
|
}
|
|
|
|
// If we have no filter, everything passes
|
|
if (!("filters" in device)) {
|
|
aService.deviceID = device.id;
|
|
return true;
|
|
}
|
|
|
|
// If all the filters pass, we have a match
|
|
let failed = false;
|
|
let filters = device.filters;
|
|
for (let filter in filters) {
|
|
if (filter in aService && aService[filter] != filters[filter]) {
|
|
failed = true;
|
|
}
|
|
}
|
|
|
|
// We found a match, so link the service to the device
|
|
if (!failed) {
|
|
aService.deviceID = device.id;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// We didn't find any matches
|
|
return false;
|
|
},
|
|
|
|
_processService: function _processService(aService) {
|
|
// Use the REST api to request more information about this service
|
|
let xhr = Cc["@mozilla.org/xmlextras/xmlhttprequest;1"].createInstance(Ci.nsIXMLHttpRequest);
|
|
xhr.open("GET", aService.location, true);
|
|
xhr.channel.loadFlags |= Ci.nsIRequest.INHIBIT_CACHING;
|
|
xhr.overrideMimeType("text/xml");
|
|
|
|
xhr.addEventListener("load", (function() {
|
|
if (xhr.status == 200) {
|
|
let doc = xhr.responseXML;
|
|
aService.appsURL = xhr.getResponseHeader("Application-URL");
|
|
if (aService.appsURL && !aService.appsURL.endsWith("/"))
|
|
aService.appsURL += "/";
|
|
aService.friendlyName = doc.querySelector("friendlyName").textContent;
|
|
aService.uuid = doc.querySelector("UDN").textContent;
|
|
aService.manufacturer = doc.querySelector("manufacturer").textContent;
|
|
aService.modelName = doc.querySelector("modelName").textContent;
|
|
|
|
this.addService(aService);
|
|
}
|
|
}).bind(this), false);
|
|
|
|
xhr.send(null);
|
|
},
|
|
|
|
// Add a service to the WeakMap, even if one already exists with this id.
|
|
// Returns true if this succeeded or false if it failed
|
|
_addService: function(service) {
|
|
// Filter out services that do not match the device filter
|
|
if (!this._filterService(service)) {
|
|
return false;
|
|
}
|
|
|
|
let device = this._devices.get(service.target);
|
|
if (device && device.mirror) {
|
|
service.mirror = true;
|
|
}
|
|
this._services.set(service.uuid, service);
|
|
return true;
|
|
},
|
|
|
|
addService: function(service) {
|
|
// Only add and notify if we don't already know about this service
|
|
if (!this._services.has(service.uuid)) {
|
|
if (!this._addService(service)) {
|
|
return;
|
|
}
|
|
Services.obs.notifyObservers(null, EVENT_SERVICE_FOUND, service.uuid);
|
|
}
|
|
|
|
// Make sure we remember this service is not stale
|
|
this._services.get(service.uuid).lastPing = this._searchTimestamp;
|
|
},
|
|
|
|
removeService: function(uuid) {
|
|
Services.obs.notifyObservers(null, EVENT_SERVICE_LOST, uuid);
|
|
this._services.delete(uuid);
|
|
},
|
|
|
|
updateService: function(service) {
|
|
if (!this._addService(service)) {
|
|
return;
|
|
}
|
|
|
|
// Make sure we remember this service is not stale
|
|
this._services.get(service.uuid).lastPing = this._searchTimestamp;
|
|
}
|
|
}
|