2011-08-10 00:02:39 +04:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is Network Location Provider for GLS.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Mozilla Foundation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2011
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Doug Turner <dougt@dougt.org>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
|
|
|
|
// Do not use this API without permission from Google.
|
|
|
|
// See http://www.google.com/support/enterprise/bin/request.py?contact_type=gme&utm_campaign=en-us-ptr-mz
|
|
|
|
// for more information.
|
|
|
|
|
2009-04-14 20:10:20 +04:00
|
|
|
Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
|
2011-08-10 00:02:39 +04:00
|
|
|
Components.utils.import("resource://gre/modules/Services.jsm");
|
2009-04-14 20:10:20 +04:00
|
|
|
|
|
|
|
const Ci = Components.interfaces;
|
|
|
|
const Cc = Components.classes;
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
let gLoggingEnabled = false;
|
|
|
|
let gTestingEnabled = false;
|
2009-04-14 20:10:20 +04:00
|
|
|
|
|
|
|
function LOG(aMsg) {
|
2009-08-15 03:12:09 +04:00
|
|
|
if (gLoggingEnabled)
|
|
|
|
{
|
2010-08-31 20:15:52 +04:00
|
|
|
aMsg = "*** WIFI GEO: " + aMsg + "\n";
|
2009-08-15 03:12:09 +04:00
|
|
|
Cc["@mozilla.org/consoleservice;1"].getService(Ci.nsIConsoleService).logStringMessage(aMsg);
|
|
|
|
dump(aMsg);
|
|
|
|
}
|
2009-04-14 20:10:20 +04:00
|
|
|
}
|
|
|
|
|
2009-08-15 03:12:09 +04:00
|
|
|
function WifiGeoCoordsObject(lat, lon, acc, alt, altacc) {
|
2011-08-10 00:02:39 +04:00
|
|
|
this.latitude = lat;
|
|
|
|
this.longitude = lon;
|
|
|
|
this.accuracy = acc;
|
|
|
|
this.altitude = alt;
|
|
|
|
this.altitudeAccuracy = altacc;
|
|
|
|
}
|
2009-04-14 20:10:20 +04:00
|
|
|
|
|
|
|
WifiGeoCoordsObject.prototype = {
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDOMGeoPositionCoords]),
|
2009-08-15 03:12:09 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
classInfo: XPCOMUtils.generateCI({interfaces: [Ci.nsIDOMGeoPositionCoords],
|
|
|
|
flags: Ci.nsIClassInfo.DOM_OBJECT,
|
|
|
|
classDescription: "wifi geo position coords object"}),
|
2009-04-14 20:10:20 +04:00
|
|
|
};
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
function WifiGeoPositionObject(lat, lng, acc) {
|
|
|
|
this.coords = new WifiGeoCoordsObject(lat, lng, acc, 0, 0);
|
|
|
|
this.address = null;
|
|
|
|
this.timestamp = Date.now();
|
|
|
|
}
|
2009-04-14 20:10:20 +04:00
|
|
|
|
|
|
|
WifiGeoPositionObject.prototype = {
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDOMGeoPosition]),
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
// Class Info is required to be able to pass objects back into the DOM.
|
|
|
|
classInfo: XPCOMUtils.generateCI({interfaces: [Ci.nsIDOMGeoPosition],
|
|
|
|
flags: Ci.nsIClassInfo.DOM_OBJECT,
|
|
|
|
classDescription: "wifi geo location position object"}),
|
2009-04-14 20:10:20 +04:00
|
|
|
};
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
function WifiGeoPositionProvider() {
|
|
|
|
try {
|
|
|
|
gLoggingEnabled = Services.prefs.getBoolPref("geo.wifi.logging.enabled");
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
try {
|
|
|
|
gTestingEnabled = Services.prefs.getBoolPref("geo.wifi.testing");
|
|
|
|
} catch (e) {}
|
|
|
|
|
2011-12-01 15:02:51 +04:00
|
|
|
this.wifiService = null;
|
|
|
|
this.timer = null;
|
|
|
|
this.hasSeenWiFi = false;
|
|
|
|
this.started = false;
|
2011-08-10 00:02:39 +04:00
|
|
|
}
|
2010-02-11 22:59:55 +03:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
WifiGeoPositionProvider.prototype = {
|
|
|
|
classID: Components.ID("{77DA64D3-7458-4920-9491-86CC9914F904}"),
|
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIGeolocationProvider,
|
|
|
|
Ci.nsIWifiListener,
|
|
|
|
Ci.nsITimerCallback]),
|
|
|
|
startup: function() {
|
|
|
|
if (this.started)
|
|
|
|
return;
|
|
|
|
this.started = true;
|
|
|
|
this.hasSeenWiFi = false;
|
|
|
|
|
|
|
|
LOG("startup called. testing mode is" + gTestingEnabled);
|
|
|
|
|
|
|
|
// if we don't see anything in 5 seconds, kick of one IP geo lookup.
|
|
|
|
// if we are testing, just hammer this callback so that we are more or less
|
|
|
|
// always sending data. It doesn't matter if we have an access point or not.
|
|
|
|
this.timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
|
|
|
|
if (!gTestingEnabled)
|
|
|
|
this.timer.initWithCallback(this, 5000, this.timer.TYPE_ONE_SHOT);
|
|
|
|
else
|
|
|
|
this.timer.initWithCallback(this, 200, this.timer.TYPE_REPEATING_SLACK);
|
|
|
|
},
|
|
|
|
|
|
|
|
watch: function(c) {
|
|
|
|
LOG("watch called");
|
|
|
|
if (!this.wifiService) {
|
|
|
|
this.wifiService = Cc["@mozilla.org/wifi/monitor;1"].getService(Components.interfaces.nsIWifiMonitor);
|
|
|
|
this.wifiService.startWatching(this);
|
2010-02-11 22:59:55 +03:00
|
|
|
}
|
2011-08-10 00:02:39 +04:00
|
|
|
},
|
2010-02-11 22:59:55 +03:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
shutdown: function() {
|
|
|
|
LOG("shutdown called");
|
|
|
|
if(this.wifiService) {
|
|
|
|
this.wifiService.stopWatching(this);
|
|
|
|
this.wifiService = null;
|
|
|
|
}
|
|
|
|
if (this.timer != null) {
|
|
|
|
this.timer.cancel();
|
|
|
|
this.timer = null;
|
2010-02-11 22:59:55 +03:00
|
|
|
}
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
// Although we aren't using cookies, we should err on the side of not
|
|
|
|
// saving any access tokens if the user asked us not to save cookies or
|
|
|
|
// has changed the lifetimePolicy. The access token in these cases is
|
|
|
|
// used and valid for the life of this object (eg. between startup and
|
|
|
|
// shutdown).
|
|
|
|
if (Services.prefs.getIntPref("network.cookie.lifetimePolicy") != 0)
|
|
|
|
Services.prefs.deleteBranch("geo.wifi.access_token.");
|
|
|
|
this.started = false;
|
|
|
|
},
|
|
|
|
|
|
|
|
getAccessTokenForURL: function(url)
|
|
|
|
{
|
|
|
|
// check to see if we have an access token:
|
|
|
|
let accessToken = "";
|
2009-07-11 04:03:48 +04:00
|
|
|
try {
|
2011-08-10 00:02:39 +04:00
|
|
|
let accessTokenPrefName = "geo.wifi.access_token." + url;
|
|
|
|
accessToken = Services.prefs.getCharPref(accessTokenPrefName);
|
|
|
|
|
|
|
|
// check to see if it has expired
|
|
|
|
let accessTokenDate = Services.prefs.getIntPref(accessTokenPrefName + ".time");
|
|
|
|
|
|
|
|
let accessTokenInterval = 1209600; // seconds in 2 weeks
|
|
|
|
try {
|
|
|
|
accessTokenInterval = Services.prefs.getIntPref("geo.wifi.access_token.recycle_interval");
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
if ((Date.now() / 1000) - accessTokenDate > accessTokenInterval)
|
|
|
|
accessToken = "";
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
accessToken = "";
|
|
|
|
}
|
|
|
|
return accessToken;
|
|
|
|
},
|
2009-08-15 03:12:09 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
onChange: function(accessPoints) {
|
|
|
|
LOG("onChange called");
|
|
|
|
this.hasSeenWiFi = true;
|
2009-08-15 03:12:09 +04:00
|
|
|
|
2011-09-27 18:23:49 +04:00
|
|
|
let providerUrlBase = "https://maps.googleapis.com/maps/api/browserlocation/json";
|
|
|
|
try {
|
|
|
|
providerUrlBase = Services.prefs.getCharPref("geo.wifi.uri");
|
|
|
|
} catch (x) {};
|
2011-08-10 00:02:39 +04:00
|
|
|
let providerUrl;
|
2009-07-11 04:03:48 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
let query = providerUrlBase.indexOf("?");
|
|
|
|
if (query == -1)
|
|
|
|
providerUrl = providerUrlBase + "?"
|
|
|
|
else
|
|
|
|
providerUrl = providerUrlBase + "&";
|
|
|
|
providerUrl = providerUrl + "browser=firefox&sensor=true";
|
|
|
|
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
let accessToken = this.getAccessTokenForURL(providerUrlBase);
|
|
|
|
if (accessToken !== "")
|
2011-11-20 15:18:27 +04:00
|
|
|
providerUrl = providerUrl + "&access_token="+accessToken;
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
function sort(a, b) {
|
|
|
|
return b.signal - a.signal;
|
|
|
|
};
|
2009-05-07 21:52:39 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
function encode(ap) {
|
|
|
|
// make sure that the ssid doesn't contain any | chars.
|
|
|
|
ap.ssid = ap.ssid.replace("|", "\\|");
|
|
|
|
// gls service parses the | as fields
|
|
|
|
return "&wifi=mac:"+ap.mac+"|ssid:"+ap.ssid+"|ss:"+ap.signal;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (accessPoints) {
|
2011-11-02 08:04:35 +04:00
|
|
|
providerUrl = providerUrl + accessPoints.sort(sort).map(encode).join("");
|
2011-08-10 00:02:39 +04:00
|
|
|
// max length is 2k. make sure we are under that
|
|
|
|
let x = providerUrl.length - 2000;
|
|
|
|
if (x >= 0) {
|
|
|
|
// we need to trim
|
|
|
|
let doomed = providerUrl.lastIndexOf("&", 2000);
|
|
|
|
LOG("Doomed:"+doomed);
|
|
|
|
providerUrl = providerUrl.substring(0, doomed);
|
2009-07-11 04:03:48 +04:00
|
|
|
}
|
2011-08-10 00:02:39 +04:00
|
|
|
}
|
2009-07-11 04:03:48 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
providerUrl = encodeURI(providerUrl);
|
|
|
|
LOG("************************************* Sending request:\n" + providerUrl + "\n");
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
// send our request to a wifi geolocation network provider:
|
|
|
|
let xhr = Components.classes["@mozilla.org/xmlextras/xmlhttprequest;1"]
|
|
|
|
.createInstance(Ci.nsIXMLHttpRequest);
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
// This is a background load
|
|
|
|
xhr.mozBackgroundRequest = true;
|
|
|
|
xhr.open("GET", providerUrl, false);
|
|
|
|
xhr.channel.loadFlags = Ci.nsIChannel.LOAD_ANONYMOUS;
|
2011-09-29 20:06:36 +04:00
|
|
|
xhr.addEventListener("error", function(req) {
|
2011-08-10 00:02:39 +04:00
|
|
|
LOG("onerror: " + req);
|
2011-09-29 20:06:36 +04:00
|
|
|
}, false);
|
|
|
|
xhr.addEventListener("load", function (req) {
|
2011-08-10 00:02:39 +04:00
|
|
|
LOG("service returned: " + req.target.responseText);
|
|
|
|
response = JSON.parse(req.target.responseText);
|
|
|
|
/*
|
|
|
|
{
|
|
|
|
"status": "OK",
|
|
|
|
"accuracy": 150.0,
|
|
|
|
"location": {
|
|
|
|
"lat": -33.85702,
|
|
|
|
"lng": 151.21494
|
|
|
|
},
|
|
|
|
"access_token": "quijibo"
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (response.status != "OK")
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (response.location) {
|
|
|
|
let newLocation = new WifiGeoPositionObject(response.location.lat,
|
|
|
|
response.location.lng,
|
|
|
|
response.accuracy);
|
|
|
|
|
|
|
|
let update = Cc["@mozilla.org/geolocation/service;1"].getService(Ci.nsIGeolocationUpdate);
|
|
|
|
update.update(newLocation);
|
|
|
|
}
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
// Check to see if we have a new access token
|
|
|
|
let newAccessToken = response.access_token;
|
|
|
|
if (newAccessToken !== undefined)
|
|
|
|
{
|
|
|
|
let accessToken = "";
|
|
|
|
let accessTokenPrefName = "geo.wifi.access_token." + providerUrlBase;
|
|
|
|
try { accessToken = Services.prefs.getCharPref(accessTokenPrefName); } catch (e) {}
|
|
|
|
|
|
|
|
if (accessToken != newAccessToken) {
|
|
|
|
// no match, lets cache
|
|
|
|
LOG("New Access Token: " + newAccessToken + "\n" + accessTokenPrefName);
|
|
|
|
try {
|
|
|
|
Services.prefs.setIntPref(accessTokenPrefName + ".time", nowInSeconds());
|
|
|
|
Services.prefs.setCharPref(accessTokenPrefName, newAccessToken);
|
|
|
|
} catch (x) {
|
|
|
|
// XXX temporary hack for bug 575346 to allow geolocation to function
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-09-29 20:06:36 +04:00
|
|
|
}, false);
|
2009-04-14 20:10:20 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
LOG("************************************* ------>>>> sending.");
|
|
|
|
xhr.send(null);
|
|
|
|
},
|
2010-08-31 20:15:52 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
onError: function (code) {
|
|
|
|
LOG("wifi error: " + code);
|
|
|
|
},
|
2010-08-31 20:15:52 +04:00
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
notify: function (timer) {
|
|
|
|
if (gTestingEnabled) {
|
|
|
|
// if we are testing, timer is repeating
|
|
|
|
this.onChange(null);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!this.hasSeenWiFi)
|
2010-08-31 20:15:52 +04:00
|
|
|
this.onChange(null);
|
2011-08-10 00:02:39 +04:00
|
|
|
this.timer = null;
|
|
|
|
}
|
|
|
|
},
|
2009-04-14 20:10:20 +04:00
|
|
|
};
|
|
|
|
|
2011-08-10 00:02:39 +04:00
|
|
|
let NSGetFactory = XPCOMUtils.generateNSGetFactory([WifiGeoPositionProvider]);
|