mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
21c98793d3
- Bug 1244883 - Add Nightly-/Aurora-only crash for AsyncTransactionWaiter timeouts - r=nical (132b2ceff9) - Bug 1148978 - Trigger paints when moving plugin windows around on the browser main thread. r=mattwoodrow (c75ce5ec09) - Dedent some functions. (bug 1254899 part 1, r=jrmuizel) (c84fb419c7) - Refactor acceleration pref initialization. (bug 1254899 part 2, r=jrmuizel) (29a164c70e) - Remove NS_NATIVE_GRAPHIC on Windows. (bug 1266536, r=jimm) (84011349d1) - Bug 1267253 - Delete gfxWindowsPlatform::RenderMode and replace it with a check against the default backend. r=bas (93cb6e503e) - Introduce gfxConfig, a manager for graphics feature settings. (bug 1254899 part 3, r=milan) (97498ca46a) - Bug 1262187: Allow D3D9 if D3D11 failed, behind the pref, but on by default. r=bas (8163e28b26) - Bug 1178376 - Optionally fade in new progressively painted tiles r=nical (777bf1799d) - Bug 1178376 - Allow progressive painting when low-precision tiles are disabled r=BenWa (3b8d84e19c) - Bug 1178376 - Put progressive paint status in tile updates r=nical (935d3b46ed) - Bug 1251778: Attempt to avoid presenting when the window is still resizing. r=jrmuizel (32b194a6f7) - Hoist mWidget into the Compositor base class. (bug 1264545 part 1, r=nical) (aca26ec343) - Lift compositor-accessed methods from nsIWidget into CompositorWidgetProxy. (bug 1264545 part 2, r=jimm) (609a23157a) - Rename FeatureStatus::Crashed to CrashedInHandler. (bug 1254899 part 4, r=milan) (0eae23a3fd) - Move DeviceInitData from gfxWindowsPlatform to gfxPlatforn. (bug 1254899 part 5, r=milan) (e31540ba18) - Merge gfxWindowsPlatform::mAcceleration into gfxConfig. (bug 1254899 part 6, r=milan) (9d45cc8b87) - Add another feature state for blacklisting and environment decisions. (bug 1254899 part 7, r=milan) (e7eee53cfb) - Give FeatureState a public interface. (bug 1254899 part 8, r=milan) (52d8e7f355) - Replace gfxWindowsPlatform::mD3D11Status with gfxConfig. (bug 1254899 part 9, r=jrmuizel) (90dc658b53) - Replace gfxWindowsPlatform::mD2D1Status with gfxConfig. (bug 1254899 part 10, r=milan) (eb9474f309) - Fix assertion failure in gfxConfig. (bug 1269565, r=milan) (0eb738ce66) - Add gfxConfig to nsIGfxInfo, for about:support access. (bug 1254899 part 11, r=jrmuizel) (e770240152) - Change Compositor::GetWidget to return a CompositorWidgetProxy. (bug 1264545 part 3, r=jimm) (fdf1d96255) - Bug 1251778 - Followup: Remove unreferenced local variable. r=bustage (27579f5542) - Use CompositorWidgetProxy in place of nsIWidget in the compositor. (bug 1264545 part 4, r=jimm) (80def1c2eb) - Use CompositorWidgetProxy in place of nsIWidget in CompositorBridgeParent. (bug 1264545 part 5, r=jimm,kats) (67d0e1ef7d) - Move CompositorWidgetProxy inheritance out of nsIWidget. (bug 1264545 part 6, r=jimm) (61075722c5) - Bug 1251894 - In CompositorD3D11::CreateTexture, copy as much as the render target allows. r=bas (bf5fc6baa2) - Bug 1266444: It is OK for us not to have texture sharing. r=jrmuizel (0b1885f89d) - Bug 1266396 - Make TextureClient more robust against racy shutdown situations. r=Bas (b1d7f54643) - Fix test bustage due to platform line-endings. Bug 1222624 (10b8cf3592) - More test bustage from bug 1222624 (763c4c0bb9) - Backed out 4 changesets (bug 1222624) to fix bug 1249572 (7ba3d433d0) - Bug 1268230 - RunTime.cpp and ScriptLoader do not have to use MainThreadStopSyncLoopRunnable, r=khuey (88499a3982) - Bug 1037725 - Add warning message in the console when worker spawn over limit. r=khuey (8af94dbc1d) - Bug 1047663 - Disabling the cache in a tab should also disable it for all workers in that tab;r=khuey (5411d81682) - Bug 1253793 Update ScriptLoader assertion to handle cancelation case. r=khuey (18c78d5651) - Bug 1245768 - Implement a test for the correct error management when worker imports 3rd party scripts, r=bz (c1d3f290a9) - Bug 1249673. Muted errors should be turned into NetworkError DOMExceptions when returning from importScripts on workers, instead of becoming NS_ERROR_FAILURE. r=baku (0358282cbe) - Bug 1265405 - Add a dictionary to specify how PeriodicWave should be normalized (or not); r=smaug …normalized (or not); r=smaugu (201213146c) - Bug 1251082. Restore comments in PageTransitionEvent.webidl that got lost when nsIDOMPageTransitionEvent.idl was migrated to webidl. r=bz The mentioned migration happened in http://hg.mozilla.org/mozilla-central/rev/e6377ca32f3d from bug 1031051. (2dfa309056) - Bug 1266178 Make ServiceWorkerClient not assert if the document doesn't have an outer window. r=ehsan (eafb169c91) - Bug 1259164 - Set ServiceWorkerMessageEvent.origin correctly when calling ServiceWorkerClient.postMessage(); r=bkelly (caeb65d10e) - Bug 1246319 P1 Dedupe service worker registrar entries. r=baku (b76deef941) - Bug 1246319 P2 Verify entries are deduped from the ServiceWorkerRegistrar. r=baku (8a4e348d6e) - Bug 1246319 P3 Fix service worker registry value update. r=bz (14abf6b6ce) - Bug 1247970 - Remove principal spec from service worker registrar file. r=baku (3c30130700) - Bug 1249438 P1 Move guts of RegisterServiceWorker() into a protected method that can be tested in gtest. r=baku (488243196d) - Bug 1249438 P2 Modify existing gtest to use RegisterServiceWorkerInternal. r=baku (e86c66891d) - Bug 1249438 P3 Add a gtest that registers duplicate service worker registrations. r=baku (35e269f9af) - Bug 1226443 P6 Ignore update() called during top level service worker script evaluation. r=ehsan (dcb9d02553) - Bug 1241725 - about:serviceworkers "Active Cache Name" UUID should not contain null bytes, r=bkelly (4cddea6a67) - Bug 1221852 - SharedWorker.port should be always not null, r=smaug (a9800274dc) - Bug 1261428: Clean up a bit more. r=bz (4977e3d7a5) - Bug 890284. Stop splitting textnodes in the XML content sink. r=peterv (a46dfca1cf) - Bug 1211708 Allow themes to specify XBL bindings even in unprivileged documents r=sicking (82cf1a4023) - Bug 915962 - Part 1: Allow pressing space to scroll the document if an editable element or form control is not focused; r=roc (cdb934af03) - Bug 915962 - Part 2: Add a test case for pressing space when a tabindex=-1 and a button element is focused; r=roc (17dcf5cfd0) - Bug 915962 - Part 3: Do not crash when pressing the space bar without having an element focused; r=roc (2161e62bc3) - Bug 1180761, cancel the event earlier so that space doesn't trigger checkbox change and scroll, r=neil (2425cb76ad) - Bug 1259182 - Shrink keyCodeData. r=bz. (737204af84) - Bug 1193567 - Check result of ReadID in nsXBLPrototypeBinding::Read(). r=wchen (c9b1c35bf3) - Bug 1173344 - Remove an intermediary root from nsXBLProtoImplField's FieldGetterImpl; r=jandem (5f42dd2e48) - Bug 1207494 - Part 14: Remove use of expression closure from dom/xbl/. r=bz (21c7d3825f) - align tests (fe34b613d3) - Bug 1223702 - Fix some errors about wifi direct. r=hchang (568d86054a) - Bug 1166274 - Part 1: Handle the callback and IPC message of setStaticIpMode correctly. r=vchang (8fb8d7f3b7) - Bug 1133665 - [Flame][Wifi] The SSID that has set to be binded with MAC address is not hightlighted when user taps it. r=hchang (3165471d13) - Bug 1207494 - Part 13: Remove use of expression closure from dom/wifi/. r=henry (dd9ad23a8a) - Bug 1251856 - Disable U2F in all releases (fix for 1231681). r=baku (24ada10566) - align tests (dae9ecd0ee) - var-let (11a3cb0878) - Bug 1184822 - Use classId to get provider. r=fabrice (1288eccd06) - Bug 1247410 - "test for _nomap ids does not work correctly". r=dougt (f736a04f08) - Bug 1035097 - Changed the type from 'radio' to 'radioType'. r=jdm (f9a0079152) - Bug 1177871 - Add a timeout to XHR request of WifiGeoPositionProvider. r=jdm (2f6aa87c20) - align code to 978593 with POST and location structure (d8ba75a759) - Bug 1230685 - Replace function declarations with add_task statements in test_storage_value_array.js and test_unicode.js. r=mak (9822bf2215) - Bug 1230683 - Replace try/catch with Assert.throws in test_storage_connection.js. r=mak (1c993fc37e) - Bug 655722 - Rewrite _buildGUIDMap in the sync bookmark engine to use PlacesUtils.promiseBookmarksTree. r=mak (3795d26af0) - Bug 1251057 - enable debug logging for rest.js requests and responses. r=adw (1062bcd113) - Bug 503515 - Try and ensure exported certificates include an extension by default. r=keeler (505967ab7f) - Bug 1017616 - Filter out some more unnecessary characters when exporting certs. r=keeler (e95838e362) - Bug 1241614 - don't overflow:auto the container, use em to size the dialog to avoid hidpi visibility issues, r=dolske,ttaubert (8bd6c2b35b) - Bug 1266851. Make <xmp> and <listing> use HTMLPreElement as their primary interface, per <whatwg/html#1015>. r=peterv (6be7f9d6e9) - Bug 1262184 - Block embed content loading when child of media element; r=bz (a297eeb378) - Bug 1263696 - Block embed content loading when ancestor of object element with content; r=bz (89c143cbfe) - Bug 1266077. Fix <base> href getter to follow the spec; it should be using the fallback base URI to resolve against, not the document URI. r=bkelly (e757b23a14) - Bug 1168079 nsTextEditRules::CollapseSelectionToTrailingBRIfNeeded() should ensure that there is a selection before calling nsEditor::GetStartNodeAndOffset() r=ehsan (6c283bf3a7) - Bug 898321 - Return success from nsTableEditor::GetCellAt if frame not found; r=ehsan (0d09143b95) - Bug 387687 - wrap quotes in plain text replies to window. r=masayuki (ca51437018) - Bug 1247483 - Only replace nodes in nsHTMLEditor::ReplaceOrphanedStructure if all nodes in node list are descendants of replacement node. r=ehsan (8416037da2) - bug 1266496 - fire some selection events for proxied accessibles r=davidb (8806de7dd9) - bug 1266518 - add a new event message for AccSelChangeEvents r=davidb (46af183cab) - Bug 953265: Adjust Opus bitrate in WebRTC to pass >8KHz audio, and comment r=bwc (b0be6a326e) - Bug 1221473: Do not treat answer as authoritative wrt payload types. r=drno (d27409209e) - Bug 1241321 - No RTCP stats for audio streams. r=rjesup (ec0222694e) - bug 1250492 - use tl::Max instead of std::max to get rid of a static constructor r=jesup (3cebbc8969) - Bug 1254187: Fix maxBitrate to respect simulcast. r=jesup (e569e54b57) - Bug 1158931 - Fix static assertion compilation error; r=snorp (eb27881746) - No bug, fix WebrtcMediaCodecVP8VideoCodec.cpp warnings (a983544581) - Bug 1252737 - use size_t instead of uint32_t for InitEncode(). r=jesup (57c3abc9fa) - Bug 1208371 - Never send more than one disabled frame in a row to the WebRTC encoder. r=jesup (ec0c28822b) - Bug 1208371 - Do image format conversion async in MediaPipeline. r=jesup (032efec783) - Bug 1266685 - Don't pass too many frames to the MediaPipelineTransmit VideoFrameConverter. r=jesup (21774a8d25) - Bug 1266644 - Rename StreamBuffer to StreamTracks. r=jesup r=pehrsons (21906fe1f7) - Bug 1208371 - Don't treat audio chunks as mutable in MediaPipeline. r=padenot (3878ef4332) - Bug 1246310 - Let MediaPipelineReceive tracks start at 0. r=jesup (b468ff8d48) - Bug 1266644 - Rename DOMMediaStream:: CreateXXXStream to DOMMediaStream:: CreateXXXStreamAsInput. r=jesup r=pehrsons (fe4b6d70bc) - Bug 1234578: Add an assertion. r=drno, a=abillings (f1a2c8d841) - bug 1250492 - include sstream in SdpMediaSection.h instead of iostream r=jesup (110b5c2eca) - Bug 1264470 - a=identity is a long attribute, r=bwc (5848194fe9) - Bug 1256750: Remove unnecessary sscanf_s parameter on windows, and fix format string everywhere else. r=jesup (371c0db476) - Bug 1204082 - try strtoull instead. r=mt (a0313aa87c) - Bug 1113443 - reject each media type with approriate default. r=bwc (a72ff312d1) - Bug 1095793 - use mid if provided to place candidate in msection. r=bwc (2c29b21fac) - Bug 1252699 - Set WEBRTC_DETECT_ARM_NEON when optional neon is requested. r=jesup (722e2043a5) - Bug 1229475 - webrtc: Call opus tonality_analysis_init. r=jesup (1cf8cc2cd7) - Bug 1254876: assert windows recording is shut down r=pkerr (1f2cb69073) - Bug 1227481 - added a memset on aec. r=jesup (532026ce20) - Bug 1254507 - Fix leak in WebRTC DesktopApplication class. r=jesup (54da72aeb4) - Bug 1196542 - share only windows with non-zero area. r=pkerr (94595ec463) - Bug 1202087 - Filter out non-shareable application for win8 or greater. r=jesup (d989956802) - Bug 1216529 - WebRTC: Request camera permission before accessing camera APIs. r=gcp (24b6699226) - Bug 1237630 - Part 1: Video freeze from WebRTC sender. r=rjesup (02daa8b5b7) - Bug 1237630 - Part 2: remove LOG statement generating a now defunct error condition. r=rjesup (c6002ef12f) - Bug 1248335: avoid using SvcInternal structure entirely, as system-vpx may not have it r=pkerr (ef9b21f20c) - Bug 1234571: unregister encoded-frame callback when releasing codec databases r=pkerr (321bd5166b) - Bug 820972 - Comment out colorTable[] because we don't need it. r=jesup. (60b10803d5) - cleanup and missing test stuff (e2be0331d9) - Merge remote-tracking branch 'upstream/dev' into winbuild (3a3bb0b315) - layout/media: fix export symbol list, fix build bustage (f9f5bfe14c)
769 lines
24 KiB
JavaScript
769 lines
24 KiB
JavaScript
/* 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/. */
|
|
|
|
const TYPE_WIFI = "wifi";
|
|
const TYPE_BLUETOOTH = "bt";
|
|
const TYPE_USB = "usb";
|
|
|
|
/**
|
|
* General tethering setting.
|
|
*/
|
|
const TETHERING_SETTING_IP = "192.168.1.1";
|
|
const TETHERING_SETTNG_PREFIX = "24";
|
|
const TETHERING_SETTING_START_IP = "192.168.1.10";
|
|
const TETHERING_SETTING_END_IP = "192.168.1.30";
|
|
const TETHERING_SETTING_DNS1 = "8.8.8.8";
|
|
const TETHERING_SETTING_DNS2 = "8.8.4.4";
|
|
|
|
const TETHERING_NETWORK_ADDR = "192.168.1.0/24";
|
|
|
|
/**
|
|
* Wifi tethering setting.
|
|
*/
|
|
const TETHERING_SETTING_SSID = "FirefoxHotSpot";
|
|
const TETHERING_SETTING_SECURITY = "open";
|
|
const TETHERING_SETTING_KEY = "1234567890";
|
|
|
|
const SETTINGS_RIL_DATA_ENABLED = 'ril.data.enabled';
|
|
const SETTINGS_KEY_DATA_APN_SETTINGS = "ril.data.apnSettings";
|
|
|
|
// Emulate Promise.jsm semantics.
|
|
Promise.defer = function() { return new Deferred(); }
|
|
function Deferred() {
|
|
this.promise = new Promise(function(resolve, reject) {
|
|
this.resolve = resolve;
|
|
this.reject = reject;
|
|
}.bind(this));
|
|
Object.freeze(this);
|
|
}
|
|
|
|
var gTestSuite = (function() {
|
|
let suite = {};
|
|
|
|
let tetheringManager;
|
|
let pendingEmulatorShellCount = 0;
|
|
|
|
/**
|
|
* A wrapper function of "is".
|
|
*
|
|
* Calls the marionette function "is" as well as throws an exception
|
|
* if the givens values are not equal.
|
|
*
|
|
* @param value1
|
|
* Any type of value to compare.
|
|
*
|
|
* @param value2
|
|
* Any type of value to compare.
|
|
*
|
|
* @param message
|
|
* Debug message for this check.
|
|
*
|
|
*/
|
|
function isOrThrow(value1, value2, message) {
|
|
is(value1, value2, message);
|
|
if (value1 !== value2) {
|
|
throw message;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Send emulator shell command with safe guard.
|
|
*
|
|
* We should only call |finish()| after all emulator command transactions
|
|
* end, so here comes with the pending counter. Resolve when the emulator
|
|
* gives positive response, and reject otherwise.
|
|
*
|
|
* Fulfill params:
|
|
* result -- an array of emulator response lines.
|
|
* Reject params:
|
|
* result -- an array of emulator response lines.
|
|
*
|
|
* @param aCommand
|
|
* A string command to be passed to emulator through its telnet console.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function runEmulatorShellSafe(aCommand) {
|
|
let deferred = Promise.defer();
|
|
|
|
++pendingEmulatorShellCount;
|
|
runEmulatorShell(aCommand, function(aResult) {
|
|
--pendingEmulatorShellCount;
|
|
|
|
ok(true, "Emulator shell response: " + JSON.stringify(aResult));
|
|
if (Array.isArray(aResult)) {
|
|
deferred.resolve(aResult);
|
|
} else {
|
|
deferred.reject(aResult);
|
|
}
|
|
});
|
|
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Wait for timeout.
|
|
*
|
|
* Resolve when the given duration elapsed. Never reject.
|
|
*
|
|
* Fulfill params: (none)
|
|
*
|
|
* @param aTimeoutMs
|
|
* The duration after which the timeout event should occurs.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function waitForTimeout(aTimeoutMs) {
|
|
let deferred = Promise.defer();
|
|
|
|
setTimeout(function() {
|
|
deferred.resolve();
|
|
}, aTimeoutMs);
|
|
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Get mozSettings value specified by @aKey.
|
|
*
|
|
* Resolve if that mozSettings value is retrieved successfully, reject
|
|
* otherwise.
|
|
*
|
|
* Fulfill params:
|
|
* The corresponding mozSettings value of the key.
|
|
* Reject params: (none)
|
|
*
|
|
* @param aKey
|
|
* A string.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function getSettings(aKey) {
|
|
let request = navigator.mozSettings.createLock().get(aKey);
|
|
|
|
return wrapDomRequestAsPromise(request)
|
|
.then(function resolve(aEvent) {
|
|
ok(true, "getSettings(" + aKey + ") - success");
|
|
return aEvent.target.result[aKey];
|
|
}, function reject(aEvent) {
|
|
ok(false, "getSettings(" + aKey + ") - error");
|
|
throw aEvent.target.error;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Set mozSettings values.
|
|
*
|
|
* Resolve if that mozSettings value is set successfully, reject otherwise.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @param aSettings
|
|
* An object of format |{key1: value1, key2: value2, ...}|.
|
|
* @return A deferred promise.
|
|
*/
|
|
function setSettings(aSettings) {
|
|
let lock = window.navigator.mozSettings.createLock();
|
|
let request = lock.set(aSettings);
|
|
let deferred = Promise.defer();
|
|
lock.onsettingstransactionsuccess = function () {
|
|
ok(true, "setSettings(" + JSON.stringify(aSettings) + ")");
|
|
deferred.resolve();
|
|
};
|
|
lock.onsettingstransactionfailure = function (aEvent) {
|
|
ok(false, "setSettings(" + JSON.stringify(aSettings) + ")");
|
|
deferred.reject();
|
|
throw aEvent.target.error;
|
|
};
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Set mozSettings value with only one key.
|
|
*
|
|
* Resolve if that mozSettings value is set successfully, reject otherwise.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @param aKey
|
|
* A string key.
|
|
* @param aValue
|
|
* An object value.
|
|
* @param aAllowError [optional]
|
|
* A boolean value. If set to true, an error response won't be treated
|
|
* as test failure. Default: false.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function setSettings1(aKey, aValue, aAllowError) {
|
|
let settings = {};
|
|
settings[aKey] = aValue;
|
|
return setSettings(settings, aAllowError);
|
|
}
|
|
|
|
/**
|
|
* Convenient MozSettings getter for SETTINGS_KEY_DATA_APN_SETTINGS.
|
|
*/
|
|
function getDataApnSettings(aAllowError) {
|
|
return getSettings(SETTINGS_KEY_DATA_APN_SETTINGS, aAllowError);
|
|
}
|
|
|
|
/**
|
|
* Convenient MozSettings setter for SETTINGS_KEY_DATA_APN_SETTINGS.
|
|
*/
|
|
function setDataApnSettings(aApnSettings, aAllowError) {
|
|
return setSettings1(SETTINGS_KEY_DATA_APN_SETTINGS, aApnSettings, aAllowError);
|
|
}
|
|
|
|
/**
|
|
* Set 'ro.tethering.dun_required' system property to 1. Note that this is a
|
|
* 'ro' property, it can only be set once.
|
|
*/
|
|
function setTetheringDunRequired() {
|
|
return runEmulatorShellSafe(['setprop', 'ro.tethering.dun_required', '1']);
|
|
}
|
|
|
|
/**
|
|
* Wrap DOMRequest onsuccess/onerror events to Promise resolve/reject.
|
|
*
|
|
* Fulfill params: A DOMEvent.
|
|
* Reject params: A DOMEvent.
|
|
*
|
|
* @param aRequest
|
|
* A DOMRequest instance.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function wrapDomRequestAsPromise(aRequest) {
|
|
let deffered = Promise.defer();
|
|
|
|
ok(aRequest instanceof DOMRequest,
|
|
"aRequest is instanceof" + aRequest.constructor);
|
|
|
|
aRequest.onsuccess = function(aEvent) {
|
|
deffered.resolve(aEvent);
|
|
};
|
|
aRequest.onerror = function(aEvent) {
|
|
deffered.reject(aEvent);
|
|
};
|
|
|
|
return deffered.promise;
|
|
}
|
|
|
|
/**
|
|
* Wait for one named MozMobileConnection event.
|
|
*
|
|
* Resolve if that named event occurs. Never reject.
|
|
*
|
|
* Fulfill params: the DOMEvent passed.
|
|
*
|
|
* @param aEventName
|
|
* A string event name.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function waitForMobileConnectionEventOnce(aEventName, aServiceId) {
|
|
aServiceId = aServiceId || 0;
|
|
|
|
let deferred = Promise.defer();
|
|
let mobileconnection = navigator.mozMobileConnections[aServiceId];
|
|
|
|
mobileconnection.addEventListener(aEventName, function onevent(aEvent) {
|
|
mobileconnection.removeEventListener(aEventName, onevent);
|
|
|
|
ok(true, "Mobile connection event '" + aEventName + "' got.");
|
|
deferred.resolve(aEvent);
|
|
});
|
|
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Wait for RIL data being connected.
|
|
*
|
|
* This function will check |MozMobileConnection.data.connected| on
|
|
* every 'datachange' event. Resolve when |MozMobileConnection.data.connected|
|
|
* becomes the expected state. Never reject.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @param aConnected
|
|
* Boolean that indicates the desired data state.
|
|
*
|
|
* @param aServiceId [optional]
|
|
* A numeric DSDS service id. Default: 0.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function waitForRilDataConnected(aConnected, aServiceId) {
|
|
aServiceId = aServiceId || 0;
|
|
|
|
return waitForMobileConnectionEventOnce('datachange', aServiceId)
|
|
.then(function () {
|
|
let mobileconnection = navigator.mozMobileConnections[aServiceId];
|
|
if (mobileconnection.data.connected !== aConnected) {
|
|
return waitForRilDataConnected(aConnected, aServiceId);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Verify everything about routing when the wifi tethering is either on or off.
|
|
*
|
|
* We use two unix commands to verify the routing: 'netcfg' and 'ip route'.
|
|
* For now the following two things will be checked:
|
|
* 1) The default route interface should be 'rmnet0'.
|
|
* 2) wlan0 is up and its ip is set to a private subnet.
|
|
*
|
|
* We also verify iptables output as netd's NatController will execute
|
|
* $ iptables -t nat -A POSTROUTING -o rmnet0 -j MASQUERADE
|
|
*
|
|
* For tethering through dun, we use 'ip rule' to find the secondary routing
|
|
* table and look for default route on that table.
|
|
*
|
|
* Resolve when the verification is successful and reject otherwise.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: String that indicates the reason of rejection.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function verifyTetheringRouting(aEnabled, aIsDun) {
|
|
let netcfgResult = {};
|
|
let ipRouteResult = {};
|
|
let ipSecondaryRouteResult = {};
|
|
|
|
// Execute 'netcfg' and parse to |netcfgResult|, each key of which is the
|
|
// interface name and value is { ip(string) }.
|
|
function exeAndParseNetcfg() {
|
|
return runEmulatorShellSafe(['netcfg'])
|
|
.then(function (aLines) {
|
|
// Sample output:
|
|
//
|
|
// lo UP 127.0.0.1/8 0x00000049 00:00:00:00:00:00
|
|
// eth0 UP 10.0.2.15/24 0x00001043 52:54:00:12:34:56
|
|
// rmnet1 DOWN 0.0.0.0/0 0x00001002 52:54:00:12:34:58
|
|
// rmnet2 DOWN 0.0.0.0/0 0x00001002 52:54:00:12:34:59
|
|
// rmnet3 DOWN 0.0.0.0/0 0x00001002 52:54:00:12:34:5a
|
|
// wlan0 UP 192.168.1.1/24 0x00001043 52:54:00:12:34:5b
|
|
// sit0 DOWN 0.0.0.0/0 0x00000080 00:00:00:00:00:00
|
|
// rmnet0 UP 10.0.2.100/24 0x00001043 52:54:00:12:34:57
|
|
//
|
|
aLines.forEach(function (aLine) {
|
|
let tokens = aLine.split(/\s+/);
|
|
if (tokens.length < 5) {
|
|
return;
|
|
}
|
|
let ifname = tokens[0];
|
|
let ip = (tokens[2].split('/'))[0];
|
|
netcfgResult[ifname] = { ip: ip };
|
|
});
|
|
});
|
|
}
|
|
|
|
// Execute 'ip route' and parse to |ipRouteResult|, each key of which is the
|
|
// interface name and value is { src(string), default(boolean) }.
|
|
function exeAndParseIpRoute() {
|
|
return runEmulatorShellSafe(['ip', 'route'])
|
|
.then(function (aLines) {
|
|
// Sample output:
|
|
//
|
|
// 10.0.2.4 via 10.0.2.2 dev rmnet0
|
|
// 10.0.2.3 via 10.0.2.2 dev rmnet0
|
|
// 192.168.1.0/24 dev wlan0 proto kernel scope link src 192.168.1.1
|
|
// 10.0.2.0/24 dev eth0 proto kernel scope link src 10.0.2.15
|
|
// 10.0.2.0/24 dev rmnet0 proto kernel scope link src 10.0.2.100
|
|
// default via 10.0.2.2 dev rmnet0
|
|
// default via 10.0.2.2 dev eth0 metric 2
|
|
//
|
|
|
|
// Parse source ip for each interface.
|
|
aLines.forEach(function (aLine) {
|
|
let tokens = aLine.trim().split(/\s+/);
|
|
let srcIndex = tokens.indexOf('src');
|
|
if (srcIndex < 0 || srcIndex + 1 >= tokens.length) {
|
|
return;
|
|
}
|
|
let ifname = tokens[2];
|
|
let src = tokens[srcIndex + 1];
|
|
ipRouteResult[ifname] = { src: src, default: false };
|
|
});
|
|
|
|
// Parse default interfaces.
|
|
aLines.forEach(function (aLine) {
|
|
let tokens = aLine.split(/\s+/);
|
|
if (tokens.length < 2) {
|
|
return;
|
|
}
|
|
if ('default' === tokens[0]) {
|
|
let ifnameIndex = tokens.indexOf('dev');
|
|
if (ifnameIndex < 0 || ifnameIndex + 1 >= tokens.length) {
|
|
return;
|
|
}
|
|
let ifname = tokens[ifnameIndex + 1];
|
|
if (ipRouteResult[ifname]) {
|
|
ipRouteResult[ifname].default = true;
|
|
}
|
|
return;
|
|
}
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// Find MASQUERADE in POSTROUTING section. 'MASQUERADE' should be found
|
|
// when tethering is enabled. 'MASQUERADE' shouldn't be found when tethering
|
|
// is disabled.
|
|
function verifyIptables() {
|
|
return runEmulatorShellSafe(['iptables', '-t', 'nat', '-L', 'POSTROUTING'])
|
|
.then(function(aLines) {
|
|
// $ iptables -t nat -L POSTROUTING
|
|
//
|
|
// Sample output (tethering on):
|
|
//
|
|
// Chain POSTROUTING (policy ACCEPT)
|
|
// target prot opt source destination
|
|
// MASQUERADE all -- anywhere anywhere
|
|
//
|
|
let found = (function find_MASQUERADE() {
|
|
// Skip first two lines.
|
|
for (let i = 2; i < aLines.length; i++) {
|
|
if (-1 !== aLines[i].indexOf('MASQUERADE')) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
})();
|
|
|
|
if ((aEnabled && !found) || (!aEnabled && found)) {
|
|
throw 'MASQUERADE' + (found ? '' : ' not') + ' found while tethering is ' +
|
|
(aEnabled ? 'enabled' : 'disabled');
|
|
}
|
|
});
|
|
}
|
|
|
|
// Execute 'ip rule show', there must be one rule for tethering network
|
|
// address to lookup for a secondary routing table, return that table id.
|
|
function verifyIpRule() {
|
|
if (!aIsDun) {
|
|
return;
|
|
}
|
|
|
|
return runEmulatorShellSafe(['ip', 'rule', 'show'])
|
|
.then(function (aLines) {
|
|
// Sample output:
|
|
//
|
|
// 0: from all lookup local
|
|
// 32765: from 192.168.1.0/24 lookup 60
|
|
// 32766: from all lookup main
|
|
// 32767: from all lookup default
|
|
//
|
|
let tableId = (function findTableId() {
|
|
for (let i = 0; i < aLines.length; i++) {
|
|
let tokens = aLines[i].split(/\s+/);
|
|
if (-1 != tokens.indexOf(TETHERING_NETWORK_ADDR)) {
|
|
let lookupIndex = tokens.indexOf('lookup');
|
|
if (lookupIndex < 0 || lookupIndex + 1 >= tokens.length) {
|
|
return;
|
|
}
|
|
return tokens[lookupIndex + 1];
|
|
}
|
|
}
|
|
return;
|
|
})();
|
|
|
|
if ((aEnabled && !tableId) || (!aEnabled && tableId)) {
|
|
throw 'Secondary table' + (tableId ? '' : ' not') + ' found while tethering is ' +
|
|
(aEnabled ? 'enabled' : 'disabled');
|
|
}
|
|
|
|
return tableId;
|
|
});
|
|
}
|
|
|
|
// Given the table id, use 'ip rule show table <table id>' to find the
|
|
// default route on that secondary routing table.
|
|
function execAndParseSecondaryTable(aTableId) {
|
|
if (!aIsDun || !aEnabled) {
|
|
return;
|
|
}
|
|
|
|
return runEmulatorShellSafe(['ip', 'route', 'show', 'table', aTableId])
|
|
.then(function (aLines) {
|
|
// We only look for default route in secondary table.
|
|
aLines.forEach(function (aLine) {
|
|
let tokens = aLine.split(/\s+/);
|
|
if (tokens.length < 2) {
|
|
return;
|
|
}
|
|
if ('default' === tokens[0]) {
|
|
let ifnameIndex = tokens.indexOf('dev');
|
|
if (ifnameIndex < 0 || ifnameIndex + 1 >= tokens.length) {
|
|
return;
|
|
}
|
|
let ifname = tokens[ifnameIndex + 1];
|
|
ipSecondaryRouteResult[ifname] = { default: true };
|
|
return;
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
function verifyDefaultRouteAndIp(aExpectedWifiTetheringIp) {
|
|
log(JSON.stringify(ipRouteResult));
|
|
log(JSON.stringify(ipSecondaryRouteResult));
|
|
log(JSON.stringify(netcfgResult));
|
|
|
|
if (aEnabled) {
|
|
isOrThrow(ipRouteResult['rmnet0'].src, netcfgResult['rmnet0'].ip, 'rmnet0.ip');
|
|
isOrThrow(ipRouteResult['rmnet0'].default, true, 'rmnet0.default');
|
|
|
|
isOrThrow(ipRouteResult['wlan0'].src, netcfgResult['wlan0'].ip, 'wlan0.ip');
|
|
isOrThrow(ipRouteResult['wlan0'].src, aExpectedWifiTetheringIp, 'expected ip');
|
|
isOrThrow(ipRouteResult['wlan0'].default, false, 'wlan0.default');
|
|
|
|
if (aIsDun) {
|
|
isOrThrow(ipRouteResult['rmnet1'].src, netcfgResult['rmnet1'].ip, 'rmnet1.ip');
|
|
isOrThrow(ipRouteResult['rmnet1'].default, false, 'rmnet1.default');
|
|
// Dun's network default route is set on secondary routing table.
|
|
isOrThrow(ipSecondaryRouteResult['rmnet1'].default, true, 'secondary rmnet1.default');
|
|
}
|
|
}
|
|
}
|
|
|
|
return verifyIptables()
|
|
.then(verifyIpRule)
|
|
.then(tableId => execAndParseSecondaryTable(tableId))
|
|
.then(exeAndParseNetcfg)
|
|
.then(exeAndParseIpRoute)
|
|
.then(() => verifyDefaultRouteAndIp(TETHERING_SETTING_IP));
|
|
}
|
|
|
|
/**
|
|
* Request to enable/disable wifi tethering.
|
|
*
|
|
* Enable/disable wifi tethering by using setTetheringEnabled API
|
|
* Resolve when the routing is verified to set up successfully in 20 seconds. The polling
|
|
* period is 1 second.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: The error message.
|
|
*
|
|
* @param aEnabled
|
|
* Boolean that indicates to enable or disable wifi tethering.
|
|
* @param aIsDun
|
|
* Boolean that indicates whether dun is required.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function setWifiTetheringEnabled(aEnabled, aIsDun) {
|
|
let RETRY_INTERVAL_MS = 1000;
|
|
let retryCnt = 20;
|
|
|
|
let config = {
|
|
"ip" : TETHERING_SETTING_IP,
|
|
"prefix" : TETHERING_SETTNG_PREFIX,
|
|
"startIp" : TETHERING_SETTING_START_IP,
|
|
"endIp" : TETHERING_SETTING_END_IP,
|
|
"dns1" : TETHERING_SETTING_DNS1,
|
|
"dns2" : TETHERING_SETTING_DNS2,
|
|
"wifiConfig": {
|
|
"ssid" : TETHERING_SETTING_SSID,
|
|
"security" : TETHERING_SETTING_SECURITY
|
|
}
|
|
};
|
|
|
|
return tetheringManager.setTetheringEnabled(aEnabled, TYPE_WIFI, config)
|
|
.then(function waitForRoutingVerified() {
|
|
return verifyTetheringRouting(aEnabled, aIsDun)
|
|
.then(null, function onreject(aReason) {
|
|
|
|
log('verifyTetheringRouting rejected due to ' + aReason +
|
|
' (' + retryCnt + ')');
|
|
|
|
if (!retryCnt--) {
|
|
throw aReason;
|
|
}
|
|
|
|
return waitForTimeout(RETRY_INTERVAL_MS).then(waitForRoutingVerified);
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Ensure wifi is enabled/disabled.
|
|
*
|
|
* Issue a wifi enable/disable request if wifi is not in the desired state;
|
|
* return a resolved promise otherwise.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @return a resolved promise or deferred promise.
|
|
*/
|
|
function ensureWifiEnabled(aEnabled) {
|
|
let wifiManager = window.navigator.mozWifiManager;
|
|
if (wifiManager.enabled === aEnabled) {
|
|
return Promise.resolve();
|
|
}
|
|
let request = wifiManager.setWifiEnabled(aEnabled);
|
|
return wrapDomRequestAsPromise(request)
|
|
}
|
|
|
|
/**
|
|
* Ensure tethering manager exists.
|
|
*
|
|
* Check navigator property |mozTetheringManager| to ensure we could access
|
|
* tethering related function.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function ensureTetheringManager() {
|
|
let deferred = Promise.defer();
|
|
|
|
tetheringManager = window.navigator.mozTetheringManager;
|
|
|
|
if (tetheringManager instanceof MozTetheringManager) {
|
|
deferred.resolve();
|
|
} else {
|
|
log("navigator.mozTetheringManager is unavailable");
|
|
deferred.reject();
|
|
}
|
|
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Add required permissions for tethering. Never reject.
|
|
*
|
|
* The permissions required for wifi testing are 'wifi-manage' and 'settings-write'.
|
|
* Never reject.
|
|
*
|
|
* Fulfill params: (none)
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
function acquirePermission() {
|
|
let deferred = Promise.defer();
|
|
|
|
let permissions = [{ 'type': 'wifi-manage', 'allow': 1, 'context': window.document },
|
|
{ 'type': 'settings-write', 'allow': 1, 'context': window.document },
|
|
{ 'type': 'settings-read', 'allow': 1, 'context': window.document },
|
|
{ 'type': 'settings-api-write', 'allow': 1, 'context': window.document },
|
|
{ 'type': 'settings-api-read', 'allow': 1, 'context': window.document },
|
|
{ 'type': 'mobileconnection', 'allow': 1, 'context': window.document }];
|
|
|
|
SpecialPowers.pushPermissions(permissions, function() {
|
|
deferred.resolve();
|
|
});
|
|
|
|
return deferred.promise;
|
|
}
|
|
|
|
/**
|
|
* Common test routine.
|
|
*
|
|
* Start a test with the given test case chain. The test environment will be
|
|
* settled down before the test. After the test, all the affected things will
|
|
* be restored.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @param aTestCaseChain
|
|
* The test case entry point, which can be a function or a promise.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
suite.startTest = function(aTestCaseChain) {
|
|
function setUp() {
|
|
return ensureTetheringManager()
|
|
.then(acquirePermission);
|
|
}
|
|
|
|
function tearDown() {
|
|
waitFor(finish, function() {
|
|
return pendingEmulatorShellCount === 0;
|
|
});
|
|
}
|
|
|
|
return setUp()
|
|
.then(aTestCaseChain)
|
|
.then(function onresolve() {
|
|
tearDown();
|
|
}, function onreject(aReason) {
|
|
ok(false, 'Promise rejects during test' + (aReason ? '(' + aReason + ')' : ''));
|
|
tearDown();
|
|
});
|
|
};
|
|
|
|
//---------------------------------------------------
|
|
// Public test suite functions
|
|
//---------------------------------------------------
|
|
suite.ensureWifiEnabled = ensureWifiEnabled;
|
|
suite.setWifiTetheringEnabled = setWifiTetheringEnabled;
|
|
suite.getDataApnSettings = getDataApnSettings;
|
|
suite.setDataApnSettings = setDataApnSettings;
|
|
suite.setTetheringDunRequired = setTetheringDunRequired;
|
|
|
|
|
|
/**
|
|
* The common test routine for wifi tethering.
|
|
*
|
|
* Set 'ril.data.enabled' to true
|
|
* before testing and restore it afterward. It will also verify 'ril.data.enabled'
|
|
* and 'tethering.wifi.enabled' to be false in the beginning. Note that this routine
|
|
* will NOT change the state of 'tethering.wifi.enabled' so the user should enable
|
|
* than disable on his/her own. This routine will only check if tethering is turned
|
|
* off after testing.
|
|
*
|
|
* Fulfill params: (none)
|
|
* Reject params: (none)
|
|
*
|
|
* @param aTestCaseChain
|
|
* The test case entry point, which can be a function or a promise.
|
|
*
|
|
* @return A deferred promise.
|
|
*/
|
|
suite.startTetheringTest = function(aTestCaseChain) {
|
|
let oriDataEnabled;
|
|
function verifyInitialState() {
|
|
return getSettings(SETTINGS_RIL_DATA_ENABLED)
|
|
.then(enabled => initTetheringTestEnvironment(enabled));
|
|
}
|
|
|
|
function initTetheringTestEnvironment(aEnabled) {
|
|
oriDataEnabled = aEnabled;
|
|
if (aEnabled) {
|
|
return Promise.resolve();
|
|
} else {
|
|
return Promise.all([waitForRilDataConnected(true),
|
|
setSettings1(SETTINGS_RIL_DATA_ENABLED, true)]);
|
|
}
|
|
}
|
|
|
|
function restoreToInitialState() {
|
|
return setSettings1(SETTINGS_RIL_DATA_ENABLED, oriDataEnabled);
|
|
}
|
|
|
|
return suite.startTest(function() {
|
|
return verifyInitialState()
|
|
.then(aTestCaseChain)
|
|
.then(restoreToInitialState, function onreject(aReason) {
|
|
return restoreToInitialState()
|
|
.then(() => { throw aReason; }); // Re-throw the orignal reject reason.
|
|
});
|
|
});
|
|
};
|
|
|
|
return suite;
|
|
})();
|