appium-uiautomator2-driver 1.73.0 → 1.74.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -29,7 +29,7 @@ platformName | Could be set to `android`. Appium itself is not strict about this
29
29
  appium:automationName | Must always be set to `uiautomator2`. Values of `automationName` are compared case-insensitively.
30
30
  appium:deviceName | The name of the device under test (actually, it is not used to select a device under test). Consider setting `udid` for real devices and `avd` for emulators instead
31
31
  appium:platformVersion | The platform version of an emulator or a real device. This capability is used for device autodetection if `udid` is not provided
32
- appium:udid | UDID of the device to be tested. Could ve retrieved from `adb devices -l` output. If unset then the driver will try to use the first connected device. Always set this capability if you run parallel tests.
32
+ appium:udid | UDID of the device to be tested. Could be retrieved from `adb devices -l` output. If unset then the driver will try to use the first connected device. Always set this capability if you run parallel tests.
33
33
  appium:noReset | Prevents the device to be reset before the session startup if set to `true`. This means that the application under test is not going to be terminated neither its data cleaned. `false` by default
34
34
  appium:fullReset | Being set to `true` always enforces the application under test to be fully uninstalled before starting a new session. `false` by default
35
35
  appium:printPageSourceOnFindFailure | Enforces the server to dump the actual XML page source into the log if any error happens. `false` by default.
@@ -210,7 +210,7 @@ Name | Description | Example
210
210
  id | This strategy is mapped to the native UiAutomator's `By.res` [locator](https://developer.android.com/reference/androidx/test/uiautomator/BySelector#res(java.lang.String)) (exact match of element's resource name). Package identifier prefix is added automatically if unset and is equal to the identifier of the current application under test. | 'com.mycompany:id/resourceId'
211
211
  accessibilityId | This strategy is mapped to the native UiAutomator's `By.desc` [locator](https://developer.android.com/reference/androidx/test/uiautomator/BySelector#desc(java.lang.String)) (exact match of element's content description). | 'my description'
212
212
  className | This strategy is mapped to the native UiAutomator's `By.clazz` [locator](https://developer.android.com/reference/androidx/test/uiautomator/BySelector#clazz(java.lang.String)) (exact match of element's class). | 'android.view.View'
213
- android uiautomator | This strategy is mapped to the native UiAutomator's `UiSelector` [locator](https://developer.android.com/reference/androidx/test/uiautomator/UiSelector)). It is even possible to perform some advanced operations, like scrolling, with this locator type | `new UiScrollable(new UiSelector().resourceId(\"android:id/list\")).scrollIntoView(new UiSelector().text(\"Radio Group\"))`
213
+ `-android uiautomator` | This strategy is mapped to the native UiAutomator's `UiSelector` [locator](https://developer.android.com/reference/androidx/test/uiautomator/UiSelector)). It is even possible to perform some advanced operations, like scrolling, with this locator type | `new UiScrollable(new UiSelector().resourceId(\"android:id/list\")).scrollIntoView(new UiSelector().text(\"Radio Group\"))`
214
214
  xpath | For elements lookup Xpath strategy the driver uses the same XML tree that is generated by page source API. Only Xpath 1.0 is supported for appium-uiatomator2-server versions below 4.25.0. All server versions starting from 4.25.0 support both Xpath 1.0 and 2.0 | `By.xpath("//android.view.View[@text=\"Regular\" and @checkable=\"true\"]")`
215
215
 
216
216
 
@@ -564,14 +564,15 @@ text | string | yes | The text to type | testing
564
564
 
565
565
  ### mobile: sensorSet
566
566
 
567
- Emulate sensors values on the connected emulator.
567
+ Emulate changing of sensor values on the connected emulator.
568
+ This extension does not work on real devices.
568
569
 
569
570
  #### Arguments
570
571
 
571
572
  Name | Type | Required | Description | Example
572
573
  --- | --- | --- | --- | ---
573
- sensorType | string | yes | Supported sensor types are: `acceleration`, `light`, `proximity`, `temperature`, `pressure` and `humidity` | light
574
- value | string | yes | value to set to the sensor | 50
574
+ sensorType | string | yes | The set of all supported sensor types could be found in [adb-emu-commands.js](https://github.com/appium/appium-adb/blob/master/lib/tools/adb-emu-commands.js) (look for *SENSORS* object values). Check the output of `sensor status` command in the [emulator console](https://developer.android.com/studio/run/emulator-console) to see more details on the available sensor types | light
575
+ value | string | yes | Check the output of `sensor get <sensorType>` command in the [emulator console](https://developer.android.com/studio/run/emulator-console) to see the acceptable value format for the given sensor type | 50
575
576
 
576
577
  ### mobile: deleteFile
577
578
 
@@ -761,6 +762,24 @@ type | string | yes | The unlock type. See the documentation on [appium:unlockTy
761
762
  strategy | string | no | Unlock strategy. See the documentation on [appium:unlockStrategy](#device-locking) capability for more details | uiautomator
762
763
  timeoutMs | number | no | Unlock timeout. See the documentation on [appium:unlockSuccessTimeout](#device-locking) capability for more details | 5000
763
764
 
765
+ ### mobile: refreshGpsCache
766
+
767
+ Sends a request to refresh the GPS cache on the device under test.
768
+ By default the location tracking is configured for
769
+ [low battery consumption](https://github.com/appium/io.appium.settings/blob/master/app/src/main/java/io/appium/settings/LocationTracker.java),
770
+ so you might need to call this extension periodically to get the updated geo
771
+ location if the actual (or mocked) device location is changed too frequently.
772
+ The feature only works if the device under test has Google Play Services installed.
773
+ In case the vanilla
774
+ [LocationManager](https://developer.android.com/reference/android/location/LocationManager)
775
+ is used the device API level must be at version 30 (Android R) or higher.
776
+
777
+ #### Arguments
778
+
779
+ Name | Type | Required | Description | Example
780
+ --- | --- | --- | --- | ---
781
+ timeoutMs | number | no | The maximum number of milliseconds to block until GPS cache is refreshed. If the API call does not receive a confirmation about successful cache refresh within this timeout then an error is thrown. Providing zero or a negative value to it skips waiting completely and does not check for any errors. 20000 ms by default. | 60000
782
+
764
783
  ## Applications Management
765
784
 
766
785
  UiAutomator2 driver supports Appium endpoints for applications management:
@@ -11,8 +11,7 @@ require("source-map-support/register");
11
11
 
12
12
  var _cssConverter = _interopRequireDefault(require("../css-converter"));
13
13
 
14
- let helpers = {},
15
- extensions = {};
14
+ const helpers = {};
16
15
  exports.helpers = helpers;
17
16
  const MAGIC_FIRST_VIS_CHILD_SEL = /\/\*\[@firstVisible ?= ?('|")true\1\]/;
18
17
  const MAGIC_SCROLLABLE_SEL = /\/\/\*\[@scrollable ?= ?('|")true\1\]/;
@@ -31,19 +30,14 @@ helpers.doFindElementOrEls = async function (params) {
31
30
 
32
31
  if (params.strategy === 'css selector') {
33
32
  params.strategy = '-android uiautomator';
34
- params.selector = _cssConverter.default.toUiAutomatorSelector(params.selector);
33
+ params.selector = new _cssConverter.default(params.selector, this.opts.appPackage).toUiAutomatorSelector();
35
34
  }
36
35
 
37
- if (params.multiple) {
38
- return await this.uiautomator2.jwproxy.command(`/elements`, 'POST', params);
39
- } else {
40
- return await this.uiautomator2.jwproxy.command(`/element`, 'POST', params);
41
- }
36
+ return await this.uiautomator2.jwproxy.command(`/element${params.multiple ? 's' : ''}`, 'POST', params);
42
37
  };
43
38
 
44
- Object.assign(extensions, helpers);
45
- var _default = extensions;
39
+ var _default = helpers;
46
40
  exports.default = _default;require('source-map-support').install();
47
41
 
48
42
 
49
- //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9jb21tYW5kcy9maW5kLmpzIl0sIm5hbWVzIjpbImhlbHBlcnMiLCJleHRlbnNpb25zIiwiTUFHSUNfRklSU1RfVklTX0NISUxEX1NFTCIsIk1BR0lDX1NDUk9MTEFCTEVfU0VMIiwiTUFHSUNfU0NST0xMQUJMRV9CWSIsImRvRmluZEVsZW1lbnRPckVscyIsInBhcmFtcyIsInN0cmF0ZWd5IiwidGVzdCIsInNlbGVjdG9yIiwiZWxlbWVudElkIiwiY29udGV4dCIsInVpYXV0b21hdG9yMiIsImp3cHJveHkiLCJjb21tYW5kIiwiQ3NzQ29udmVydGVyIiwidG9VaUF1dG9tYXRvclNlbGVjdG9yIiwibXVsdGlwbGUiLCJPYmplY3QiLCJhc3NpZ24iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7O0FBRUEsSUFBSUEsT0FBTyxHQUFHLEVBQWQ7QUFBQSxJQUFrQkMsVUFBVSxHQUFHLEVBQS9COztBQUlBLE1BQU1DLHlCQUF5QixHQUFHLHVDQUFsQztBQUVBLE1BQU1DLG9CQUFvQixHQUFHLHVDQUE3QjtBQUNBLE1BQU1DLG1CQUFtQixHQUFHLG1DQUE1Qjs7QUFNQUosT0FBTyxDQUFDSyxrQkFBUixHQUE2QixnQkFBZ0JDLE1BQWhCLEVBQXdCO0FBQ25ELE1BQUlBLE1BQU0sQ0FBQ0MsUUFBUCxLQUFvQixPQUFwQixJQUErQkwseUJBQXlCLENBQUNNLElBQTFCLENBQStCRixNQUFNLENBQUNHLFFBQXRDLENBQW5DLEVBQW9GO0FBQ2xGLFFBQUlDLFNBQVMsR0FBR0osTUFBTSxDQUFDSyxPQUF2QjtBQUNBLFdBQU8sTUFBTSxLQUFLQyxZQUFMLENBQWtCQyxPQUFsQixDQUEwQkMsT0FBMUIsQ0FBbUMsbUJBQWtCSixTQUFVLGdCQUEvRCxFQUFnRixLQUFoRixFQUF1RixFQUF2RixDQUFiO0FBQ0Q7O0FBQ0QsTUFBSUosTUFBTSxDQUFDQyxRQUFQLEtBQW9CLE9BQXBCLElBQStCSixvQkFBb0IsQ0FBQ0ssSUFBckIsQ0FBMEJGLE1BQU0sQ0FBQ0csUUFBakMsQ0FBbkMsRUFBK0U7QUFDN0VILElBQUFBLE1BQU0sQ0FBQ0MsUUFBUCxHQUFrQixzQkFBbEI7QUFDQUQsSUFBQUEsTUFBTSxDQUFDRyxRQUFQLEdBQWtCTCxtQkFBbEI7QUFDRDs7QUFDRCxNQUFJRSxNQUFNLENBQUNDLFFBQVAsS0FBb0IsY0FBeEIsRUFBd0M7QUFDdENELElBQUFBLE1BQU0sQ0FBQ0MsUUFBUCxHQUFrQixzQkFBbEI7QUFDQUQsSUFBQUEsTUFBTSxDQUFDRyxRQUFQLEdBQWtCTSxzQkFBYUMscUJBQWIsQ0FBbUNWLE1BQU0sQ0FBQ0csUUFBMUMsQ0FBbEI7QUFDRDs7QUFDRCxNQUFJSCxNQUFNLENBQUNXLFFBQVgsRUFBcUI7QUFDbkIsV0FBTyxNQUFNLEtBQUtMLFlBQUwsQ0FBa0JDLE9BQWxCLENBQTBCQyxPQUExQixDQUFtQyxXQUFuQyxFQUErQyxNQUEvQyxFQUF1RFIsTUFBdkQsQ0FBYjtBQUNELEdBRkQsTUFFTztBQUNMLFdBQU8sTUFBTSxLQUFLTSxZQUFMLENBQWtCQyxPQUFsQixDQUEwQkMsT0FBMUIsQ0FBbUMsVUFBbkMsRUFBOEMsTUFBOUMsRUFBc0RSLE1BQXRELENBQWI7QUFDRDtBQUNGLENBbEJEOztBQW9CQVksTUFBTSxDQUFDQyxNQUFQLENBQWNsQixVQUFkLEVBQTBCRCxPQUExQjtlQUVlQyxVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IENzc0NvbnZlcnRlciBmcm9tICcuLi9jc3MtY29udmVydGVyJztcblxubGV0IGhlbHBlcnMgPSB7fSwgZXh0ZW5zaW9ucyA9IHt9O1xuXG4vLyB3ZSBvdmVycmlkZSB0aGUgeHBhdGggc2VhcmNoIGZvciB0aGlzIGZpcnN0LXZpc2libGUtY2hpbGQgc2VsZWN0b3IsIHdoaWNoXG4vLyBsb29rcyBsaWtlIC8qW0BmaXJzdFZpc2libGU9XCJ0cnVlXCJdXG5jb25zdCBNQUdJQ19GSVJTVF9WSVNfQ0hJTERfU0VMID0gL1xcL1xcKlxcW0BmaXJzdFZpc2libGUgPz0gPygnfFwiKXRydWVcXDFcXF0vO1xuXG5jb25zdCBNQUdJQ19TQ1JPTExBQkxFX1NFTCA9IC9cXC9cXC9cXCpcXFtAc2Nyb2xsYWJsZSA/PSA/KCd8XCIpdHJ1ZVxcMVxcXS87XG5jb25zdCBNQUdJQ19TQ1JPTExBQkxFX0JZID0gJ25ldyBVaVNlbGVjdG9yKCkuc2Nyb2xsYWJsZSh0cnVlKSc7XG5cbi8qKlxuICogT3ZlcnJpZGluZyBoZWxwZXJzLmRvRmluZEVsZW1lbnRPckVscyBmdW5jdGlvbmFsaXR5IG9mIGFwcGl1bS1hbmRyb2lkLWRyaXZlcixcbiAqIHRoaXMuZWxlbWVudCBpbml0aWFsaXplZCBpbiBmaW5kLmpzIG9mIGFwcGl1bS1hbmRyb2lkLWRyaXZlLlxuICovXG5oZWxwZXJzLmRvRmluZEVsZW1lbnRPckVscyA9IGFzeW5jIGZ1bmN0aW9uIChwYXJhbXMpIHtcbiAgaWYgKHBhcmFtcy5zdHJhdGVneSA9PT0gJ3hwYXRoJyAmJiBNQUdJQ19GSVJTVF9WSVNfQ0hJTERfU0VMLnRlc3QocGFyYW1zLnNlbGVjdG9yKSkge1xuICAgIGxldCBlbGVtZW50SWQgPSBwYXJhbXMuY29udGV4dDtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy51aWF1dG9tYXRvcjIuandwcm94eS5jb21tYW5kKGAvYXBwaXVtL2VsZW1lbnQvJHtlbGVtZW50SWR9L2ZpcnN0X3Zpc2libGVgLCAnR0VUJywge30pO1xuICB9XG4gIGlmIChwYXJhbXMuc3RyYXRlZ3kgPT09ICd4cGF0aCcgJiYgTUFHSUNfU0NST0xMQUJMRV9TRUwudGVzdChwYXJhbXMuc2VsZWN0b3IpKSB7XG4gICAgcGFyYW1zLnN0cmF0ZWd5ID0gJy1hbmRyb2lkIHVpYXV0b21hdG9yJztcbiAgICBwYXJhbXMuc2VsZWN0b3IgPSBNQUdJQ19TQ1JPTExBQkxFX0JZO1xuICB9XG4gIGlmIChwYXJhbXMuc3RyYXRlZ3kgPT09ICdjc3Mgc2VsZWN0b3InKSB7XG4gICAgcGFyYW1zLnN0cmF0ZWd5ID0gJy1hbmRyb2lkIHVpYXV0b21hdG9yJztcbiAgICBwYXJhbXMuc2VsZWN0b3IgPSBDc3NDb252ZXJ0ZXIudG9VaUF1dG9tYXRvclNlbGVjdG9yKHBhcmFtcy5zZWxlY3Rvcik7XG4gIH1cbiAgaWYgKHBhcmFtcy5tdWx0aXBsZSkge1xuICAgIHJldHVybiBhd2FpdCB0aGlzLnVpYXV0b21hdG9yMi5qd3Byb3h5LmNvbW1hbmQoYC9lbGVtZW50c2AsICdQT1NUJywgcGFyYW1zKTtcbiAgfSBlbHNlIHtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy51aWF1dG9tYXRvcjIuandwcm94eS5jb21tYW5kKGAvZWxlbWVudGAsICdQT1NUJywgcGFyYW1zKTtcbiAgfVxufTtcblxuT2JqZWN0LmFzc2lnbihleHRlbnNpb25zLCBoZWxwZXJzKTtcbmV4cG9ydCB7IGhlbHBlcnMgfTtcbmV4cG9ydCBkZWZhdWx0IGV4dGVuc2lvbnM7XG4iXSwiZmlsZSI6ImxpYi9jb21tYW5kcy9maW5kLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
43
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImxpYi9jb21tYW5kcy9maW5kLmpzIl0sIm5hbWVzIjpbImhlbHBlcnMiLCJNQUdJQ19GSVJTVF9WSVNfQ0hJTERfU0VMIiwiTUFHSUNfU0NST0xMQUJMRV9TRUwiLCJNQUdJQ19TQ1JPTExBQkxFX0JZIiwiZG9GaW5kRWxlbWVudE9yRWxzIiwicGFyYW1zIiwic3RyYXRlZ3kiLCJ0ZXN0Iiwic2VsZWN0b3IiLCJlbGVtZW50SWQiLCJjb250ZXh0IiwidWlhdXRvbWF0b3IyIiwiandwcm94eSIsImNvbW1hbmQiLCJDc3NDb252ZXJ0ZXIiLCJvcHRzIiwiYXBwUGFja2FnZSIsInRvVWlBdXRvbWF0b3JTZWxlY3RvciIsIm11bHRpcGxlIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7OztBQUFBOztBQUVBLE1BQU1BLE9BQU8sR0FBRyxFQUFoQjs7QUFJQSxNQUFNQyx5QkFBeUIsR0FBRyx1Q0FBbEM7QUFFQSxNQUFNQyxvQkFBb0IsR0FBRyx1Q0FBN0I7QUFDQSxNQUFNQyxtQkFBbUIsR0FBRyxtQ0FBNUI7O0FBTUFILE9BQU8sQ0FBQ0ksa0JBQVIsR0FBNkIsZ0JBQWdCQyxNQUFoQixFQUF3QjtBQUNuRCxNQUFJQSxNQUFNLENBQUNDLFFBQVAsS0FBb0IsT0FBcEIsSUFBK0JMLHlCQUF5QixDQUFDTSxJQUExQixDQUErQkYsTUFBTSxDQUFDRyxRQUF0QyxDQUFuQyxFQUFvRjtBQUNsRixRQUFJQyxTQUFTLEdBQUdKLE1BQU0sQ0FBQ0ssT0FBdkI7QUFDQSxXQUFPLE1BQU0sS0FBS0MsWUFBTCxDQUFrQkMsT0FBbEIsQ0FBMEJDLE9BQTFCLENBQW1DLG1CQUFrQkosU0FBVSxnQkFBL0QsRUFBZ0YsS0FBaEYsRUFBdUYsRUFBdkYsQ0FBYjtBQUNEOztBQUNELE1BQUlKLE1BQU0sQ0FBQ0MsUUFBUCxLQUFvQixPQUFwQixJQUErQkosb0JBQW9CLENBQUNLLElBQXJCLENBQTBCRixNQUFNLENBQUNHLFFBQWpDLENBQW5DLEVBQStFO0FBQzdFSCxJQUFBQSxNQUFNLENBQUNDLFFBQVAsR0FBa0Isc0JBQWxCO0FBQ0FELElBQUFBLE1BQU0sQ0FBQ0csUUFBUCxHQUFrQkwsbUJBQWxCO0FBQ0Q7O0FBQ0QsTUFBSUUsTUFBTSxDQUFDQyxRQUFQLEtBQW9CLGNBQXhCLEVBQXdDO0FBQ3RDRCxJQUFBQSxNQUFNLENBQUNDLFFBQVAsR0FBa0Isc0JBQWxCO0FBQ0FELElBQUFBLE1BQU0sQ0FBQ0csUUFBUCxHQUFrQixJQUFJTSxxQkFBSixDQUFpQlQsTUFBTSxDQUFDRyxRQUF4QixFQUFrQyxLQUFLTyxJQUFMLENBQVVDLFVBQTVDLEVBQ2ZDLHFCQURlLEVBQWxCO0FBRUQ7O0FBQ0QsU0FBTyxNQUFNLEtBQUtOLFlBQUwsQ0FBa0JDLE9BQWxCLENBQTBCQyxPQUExQixDQUFtQyxXQUFVUixNQUFNLENBQUNhLFFBQVAsR0FBa0IsR0FBbEIsR0FBd0IsRUFBRyxFQUF4RSxFQUEyRSxNQUEzRSxFQUFtRmIsTUFBbkYsQ0FBYjtBQUNELENBZkQ7O2VBa0JlTCxPIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IENzc0NvbnZlcnRlciBmcm9tICcuLi9jc3MtY29udmVydGVyJztcblxuY29uc3QgaGVscGVycyA9IHt9O1xuXG4vLyB3ZSBvdmVycmlkZSB0aGUgeHBhdGggc2VhcmNoIGZvciB0aGlzIGZpcnN0LXZpc2libGUtY2hpbGQgc2VsZWN0b3IsIHdoaWNoXG4vLyBsb29rcyBsaWtlIC8qW0BmaXJzdFZpc2libGU9XCJ0cnVlXCJdXG5jb25zdCBNQUdJQ19GSVJTVF9WSVNfQ0hJTERfU0VMID0gL1xcL1xcKlxcW0BmaXJzdFZpc2libGUgPz0gPygnfFwiKXRydWVcXDFcXF0vO1xuXG5jb25zdCBNQUdJQ19TQ1JPTExBQkxFX1NFTCA9IC9cXC9cXC9cXCpcXFtAc2Nyb2xsYWJsZSA/PSA/KCd8XCIpdHJ1ZVxcMVxcXS87XG5jb25zdCBNQUdJQ19TQ1JPTExBQkxFX0JZID0gJ25ldyBVaVNlbGVjdG9yKCkuc2Nyb2xsYWJsZSh0cnVlKSc7XG5cbi8qKlxuICogT3ZlcnJpZGluZyBoZWxwZXJzLmRvRmluZEVsZW1lbnRPckVscyBmdW5jdGlvbmFsaXR5IG9mIGFwcGl1bS1hbmRyb2lkLWRyaXZlcixcbiAqIHRoaXMuZWxlbWVudCBpbml0aWFsaXplZCBpbiBmaW5kLmpzIG9mIGFwcGl1bS1hbmRyb2lkLWRyaXZlLlxuICovXG5oZWxwZXJzLmRvRmluZEVsZW1lbnRPckVscyA9IGFzeW5jIGZ1bmN0aW9uIChwYXJhbXMpIHtcbiAgaWYgKHBhcmFtcy5zdHJhdGVneSA9PT0gJ3hwYXRoJyAmJiBNQUdJQ19GSVJTVF9WSVNfQ0hJTERfU0VMLnRlc3QocGFyYW1zLnNlbGVjdG9yKSkge1xuICAgIGxldCBlbGVtZW50SWQgPSBwYXJhbXMuY29udGV4dDtcbiAgICByZXR1cm4gYXdhaXQgdGhpcy51aWF1dG9tYXRvcjIuandwcm94eS5jb21tYW5kKGAvYXBwaXVtL2VsZW1lbnQvJHtlbGVtZW50SWR9L2ZpcnN0X3Zpc2libGVgLCAnR0VUJywge30pO1xuICB9XG4gIGlmIChwYXJhbXMuc3RyYXRlZ3kgPT09ICd4cGF0aCcgJiYgTUFHSUNfU0NST0xMQUJMRV9TRUwudGVzdChwYXJhbXMuc2VsZWN0b3IpKSB7XG4gICAgcGFyYW1zLnN0cmF0ZWd5ID0gJy1hbmRyb2lkIHVpYXV0b21hdG9yJztcbiAgICBwYXJhbXMuc2VsZWN0b3IgPSBNQUdJQ19TQ1JPTExBQkxFX0JZO1xuICB9XG4gIGlmIChwYXJhbXMuc3RyYXRlZ3kgPT09ICdjc3Mgc2VsZWN0b3InKSB7XG4gICAgcGFyYW1zLnN0cmF0ZWd5ID0gJy1hbmRyb2lkIHVpYXV0b21hdG9yJztcbiAgICBwYXJhbXMuc2VsZWN0b3IgPSBuZXcgQ3NzQ29udmVydGVyKHBhcmFtcy5zZWxlY3RvciwgdGhpcy5vcHRzLmFwcFBhY2thZ2UpXG4gICAgICAudG9VaUF1dG9tYXRvclNlbGVjdG9yKCk7XG4gIH1cbiAgcmV0dXJuIGF3YWl0IHRoaXMudWlhdXRvbWF0b3IyLmp3cHJveHkuY29tbWFuZChgL2VsZW1lbnQke3BhcmFtcy5tdWx0aXBsZSA/ICdzJyA6ICcnfWAsICdQT1NUJywgcGFyYW1zKTtcbn07XG5cbmV4cG9ydCB7IGhlbHBlcnMgfTtcbmV4cG9ydCBkZWZhdWx0IGhlbHBlcnM7XG4iXSwiZmlsZSI6ImxpYi9jb21tYW5kcy9maW5kLmpzIiwic291cmNlUm9vdCI6Ii4uLy4uLy4uIn0=
@@ -165,7 +165,8 @@ extensions.executeMobile = async function (mobileCommand, opts = {}) {
165
165
  broadcast: 'mobileBroadcast',
166
166
  getContexts: 'mobileGetContexts',
167
167
  installMultipleApks: 'mobileInstallMultipleApks',
168
- unlock: 'mobileUnlock'
168
+ unlock: 'mobileUnlock',
169
+ refreshGpsCache: 'mobileRefreshGpsCache'
169
170
  };
170
171
 
171
172
  if (!_lodash.default.has(mobileCommandsMapping, mobileCommand)) {
@@ -278,4 +279,4 @@ var _default = extensions;
278
279
  exports.default = _default;require('source-map-support').install();
279
280
 
280
281
 
281
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/general.js"],"names":["extensions","commands","helpers","getPageSource","uiautomator2","jwproxy","command","getClipboard","adb","getApiLevel","doSendKeys","params","keyevent","keycode","metastate","log","debug","back","getStrings","language","getDeviceLanguage","info","preprocessStringsMap","mapping","result","key","value","_","toPairs","isString","JSON","stringify","apkStrings","opts","app","appPackage","errorAndThrow","tmpRoot","tempDir","openDir","pullApk","err","message","fs","exists","extractStringsFromApk","rimraf","getDisplayDensity","getWindowSize","getWindowRect","width","height","x","y","executeMobile","mobileCommand","mobileCommandsMapping","shell","execEmuConsoleCommand","dragGesture","flingGesture","doubleClickGesture","clickGesture","longClickGesture","pinchCloseGesture","pinchOpenGesture","swipeGesture","scrollGesture","scrollBackTo","scroll","viewportScreenshot","viewportRect","deepLink","startLogsBroadcast","stopLogsBroadcast","acceptAlert","dismissAlert","batteryInfo","deviceInfo","getDeviceTime","changePermissions","getPermissions","performEditorAction","startScreenStreaming","stopScreenStreaming","getNotifications","listSms","type","sensorSet","deleteFile","clearApp","startActivity","startService","stopService","broadcast","getContexts","installMultipleApks","unlock","has","errors","UnknownCommandError","keys","mobileViewportScreenshot","getViewportScreenshot","mobileViewPortRect","getViewPortRect","setUrl","url","startUri","mobileDeepLink","package","pkg","waitForLaunch","openNotifications","updateSettings","settings","driverOnlySettings","serverSettings","setting","BASEDRIVER_HANDLED_SETTINGS","includes","isEmpty","update","getSettings","wrapBootstrapDisconnect","wrapped","suspendChromedriverProxy","chromedriver","proxyReqRes","bind","proxyCommand","jwpProxyActive","mobileGetDeviceInfo","mobileType","text","isUndefined","typeUnicode","mobileInstallMultipleApks","isArray","apks","InvalidArgumentError","B","all","map","configureApp","APK_EXTENSION","options","Object","assign"],"mappings":";;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA,IAAIA,UAAU,GAAG,EAAjB;AAAA,IACIC,QAAQ,GAAG,EADf;AAAA,IAEIC,OAAO,GAAG,EAFd;;AAIAD,QAAQ,CAACE,aAAT,GAAyB,kBAAkB;AACzC,SAAO,MAAM,KAAKC,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,SAAlC,EAA6C,KAA7C,EAAoD,EAApD,CAAb;AACD,CAFD;;AAIAL,QAAQ,CAACM,YAAT,GAAwB,kBAAkB;AACxC,SAAQ,OAAM,KAAKC,GAAL,CAASC,WAAT,EAAN,IAA+B,EAAhC,GACF,MAAM,KAAKL,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,8BAAlC,EAAkE,MAAlE,EAA0E,EAA1E,CADJ,GAEF,MAAM,KAAKE,GAAL,CAASD,YAAT,EAFX;AAGD,CAJD;;AAOAN,QAAQ,CAACS,UAAT,GAAsB,gBAAgBC,MAAhB,EAAwB;AAC5C,QAAM,KAAKP,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,OAAlC,EAA2C,MAA3C,EAAmDK,MAAnD,CAAN;AACD,CAFD;;AAKAV,QAAQ,CAACW,QAAT,GAAoB,gBAAgBC,OAAhB,EAAyBC,SAAzB,EAAoC;AACtDC,kBAAIC,KAAJ,CAAW,sBAAqBF,SAAU,EAA1C;;AACA,QAAM,KAAKN,GAAL,CAASI,QAAT,CAAkBC,OAAlB,CAAN;AACD,CAHD;;AAMAZ,QAAQ,CAACgB,IAAT,GAAgB,kBAAkB;AAChC,QAAM,KAAKT,GAAL,CAASI,QAAT,CAAkB,CAAlB,CAAN;AACD,CAFD;;AAIAX,QAAQ,CAACiB,UAAT,GAAsB,gBAAgBC,QAAhB,EAA0B;AAC9C,MAAI,CAACA,QAAL,EAAe;AACbA,IAAAA,QAAQ,GAAG,MAAM,KAAKX,GAAL,CAASY,iBAAT,EAAjB;;AACAL,oBAAIM,IAAJ,CAAU,iDAAgDF,QAAS,EAAnE;AACD;;AAID,QAAMG,oBAAoB,GAAG,UAAUC,OAAV,EAAmB;AAC9C,UAAMC,MAAM,GAAG,EAAf;;AACA,SAAK,MAAM,CAACC,GAAD,EAAMC,KAAN,CAAX,IAA2BC,gBAAEC,OAAF,CAAUL,OAAV,CAA3B,EAA+C;AAC7CC,MAAAA,MAAM,CAACC,GAAD,CAAN,GAAcE,gBAAEE,QAAF,CAAWH,KAAX,IAAoBA,KAApB,GAA4BI,IAAI,CAACC,SAAL,CAAeL,KAAf,CAA1C;AACD;;AACD,WAAOF,MAAP;AACD,GAND;;AAQA,MAAI,KAAKQ,UAAL,CAAgBb,QAAhB,CAAJ,EAA+B;AAE7B,WAAOG,oBAAoB,CAAC,KAAKU,UAAL,CAAgBb,QAAhB,CAAD,CAA3B;AACD;;AAED,MAAI,CAAC,KAAKc,IAAL,CAAUC,GAAX,IAAkB,CAAC,KAAKD,IAAL,CAAUE,UAAjC,EAA6C;AAC3CpB,oBAAIqB,aAAJ,CAAkB,oEAAlB;AACD;;AAED,MAAIF,GAAG,GAAG,KAAKD,IAAL,CAAUC,GAApB;AACA,QAAMG,OAAO,GAAG,MAAMC,uBAAQC,OAAR,EAAtB;;AACA,MAAI;AACF,QAAI,CAACL,GAAL,EAAU;AACR,UAAI;AACFA,QAAAA,GAAG,GAAG,MAAM,KAAK1B,GAAL,CAASgC,OAAT,CAAiB,KAAKP,IAAL,CAAUE,UAA3B,EAAuCE,OAAvC,CAAZ;AACD,OAFD,CAEE,OAAOI,GAAP,EAAY;AACZ1B,wBAAIqB,aAAJ,CAAmB,+BAA8B,KAAKH,IAAL,CAAUE,UAAW,sBAAqBM,GAAG,CAACC,OAAQ,EAAvG;AACD;AACF;;AAED,QAAI,EAAC,MAAMC,kBAAGC,MAAH,CAAUV,GAAV,CAAP,CAAJ,EAA2B;AACzBnB,sBAAIqB,aAAJ,CAAmB,eAAcF,GAAI,kBAArC;AACD;;AAED,QAAI;AACF,YAAM;AAACF,QAAAA;AAAD,UAAe,MAAM,KAAKxB,GAAL,CAASqC,qBAAT,CAA+BX,GAA/B,EAAoCf,QAApC,EAA8CkB,OAA9C,CAA3B;AACA,WAAKL,UAAL,CAAgBb,QAAhB,IAA4Ba,UAA5B;AACA,aAAOV,oBAAoB,CAACU,UAAD,CAA3B;AACD,KAJD,CAIE,OAAOS,GAAP,EAAY;AACZ1B,sBAAIqB,aAAJ,CAAmB,gCAA+BF,GAAI,sBAAqBO,GAAG,CAACC,OAAQ,EAAvF;AACD;AACF,GApBD,SAoBU;AACR,UAAMC,kBAAGG,MAAH,CAAUT,OAAV,CAAN;AACD;AACF,CAlDD;;AAoDApC,QAAQ,CAAC8C,iBAAT,GAA6B,eAAeA,iBAAf,GAAoC;AAC/D,SAAO,MAAM,KAAK3C,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,gCAAlC,EAAoE,KAApE,EAA2E,EAA3E,CAAb;AACD,CAFD;;AAKAL,QAAQ,CAAC+C,aAAT,GAAyB,kBAAkB;AACzC,SAAO,MAAM,KAAK5C,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,sBAAlC,EAA0D,KAA1D,EAAiE,EAAjE,CAAb;AACD,CAFD;;AAKAL,QAAQ,CAACgD,aAAT,GAAyB,kBAAkB;AACzC,QAAM;AAACC,IAAAA,KAAD;AAAQC,IAAAA;AAAR,MAAkB,MAAM,KAAKH,aAAL,EAA9B;AACA,SAAO;AACLE,IAAAA,KADK;AAELC,IAAAA,MAFK;AAGLC,IAAAA,CAAC,EAAE,CAHE;AAILC,IAAAA,CAAC,EAAE;AAJE,GAAP;AAMD,CARD;;AAUArD,UAAU,CAACsD,aAAX,GAA2B,gBAAgBC,aAAhB,EAA+BtB,IAAI,GAAG,EAAtC,EAA0C;AACnE,QAAMuB,qBAAqB,GAAG;AAC5BC,IAAAA,KAAK,EAAE,aADqB;AAG5BC,IAAAA,qBAAqB,EAAE,6BAHK;AAK5BC,IAAAA,WAAW,EAAE,mBALe;AAM5BC,IAAAA,YAAY,EAAE,oBANc;AAO5BC,IAAAA,kBAAkB,EAAE,0BAPQ;AAQ5BC,IAAAA,YAAY,EAAE,oBARc;AAS5BC,IAAAA,gBAAgB,EAAE,wBATU;AAU5BC,IAAAA,iBAAiB,EAAE,yBAVS;AAW5BC,IAAAA,gBAAgB,EAAE,wBAXU;AAY5BC,IAAAA,YAAY,EAAE,oBAZc;AAa5BC,IAAAA,aAAa,EAAE,qBAba;AAc5BC,IAAAA,YAAY,EAAE,oBAdc;AAe5BC,IAAAA,MAAM,EAAE,cAfoB;AAgB5BC,IAAAA,kBAAkB,EAAE,0BAhBQ;AAiB5BC,IAAAA,YAAY,EAAE,oBAjBc;AAmB5BC,IAAAA,QAAQ,EAAE,gBAnBkB;AAqB5BC,IAAAA,kBAAkB,EAAE,0BArBQ;AAsB5BC,IAAAA,iBAAiB,EAAE,yBAtBS;AAwB5BC,IAAAA,WAAW,EAAE,mBAxBe;AAyB5BC,IAAAA,YAAY,EAAE,oBAzBc;AA2B5BC,IAAAA,WAAW,EAAE,sBA3Be;AA6B5BC,IAAAA,UAAU,EAAE,qBA7BgB;AA+B5BC,IAAAA,aAAa,EAAE,qBA/Ba;AAiC5BC,IAAAA,iBAAiB,EAAE,yBAjCS;AAkC5BC,IAAAA,cAAc,EAAE,sBAlCY;AAoC5BC,IAAAA,mBAAmB,EAAE,2BApCO;AAsC5BC,IAAAA,oBAAoB,EAAE,4BAtCM;AAuC5BC,IAAAA,mBAAmB,EAAE,2BAvCO;AAyC5BC,IAAAA,gBAAgB,EAAE,wBAzCU;AA2C5BC,IAAAA,OAAO,EAAE,eA3CmB;AA6C5BC,IAAAA,IAAI,EAAE,YA7CsB;AA8C5BC,IAAAA,SAAS,EAAE,WA9CiB;AAgD5BC,IAAAA,UAAU,EAAE,kBAhDgB;AAkD5BC,IAAAA,QAAQ,EAAE,gBAlDkB;AAoD5BC,IAAAA,aAAa,EAAE,qBApDa;AAqD5BC,IAAAA,YAAY,EAAE,oBArDc;AAsD5BC,IAAAA,WAAW,EAAE,mBAtDe;AAuD5BC,IAAAA,SAAS,EAAE,iBAvDiB;AAyD5BC,IAAAA,WAAW,EAAE,mBAzDe;AA2D5BC,IAAAA,mBAAmB,EAAE,2BA3DO;AA6D5BC,IAAAA,MAAM,EAAE;AA7DoB,GAA9B;;AAgEA,MAAI,CAACtE,gBAAEuE,GAAF,CAAM1C,qBAAN,EAA6BD,aAA7B,CAAL,EAAkD;AAChD,UAAM,IAAI4C,yBAAOC,mBAAX,CAAgC,2BAA0B7C,aAAc,KAAzC,GAClC,QAAO5B,gBAAE0E,IAAF,CAAO7C,qBAAP,CAA8B,0BADlC,CAAN;AAED;;AACD,SAAO,MAAM,KAAKA,qBAAqB,CAACD,aAAD,CAA1B,EAA2CtB,IAA3C,CAAb;AACD,CAtED;;AAwEAhC,QAAQ,CAACqG,wBAAT,GAAoC,kBAAkB;AACpD,SAAO,MAAM,KAAKC,qBAAL,EAAb;AACD,CAFD;;AAgBAtG,QAAQ,CAACuG,kBAAT,GAA8B,eAAeA,kBAAf,GAAqC;AACjE,SAAO,MAAM,KAAKC,eAAL,EAAb;AACD,CAFD;;AAIAxG,QAAQ,CAACyG,MAAT,GAAkB,gBAAgBC,GAAhB,EAAqB;AACrC,QAAM,KAAKnG,GAAL,CAASoG,QAAT,CAAkBD,GAAlB,EAAuB,KAAK1E,IAAL,CAAUE,UAAjC,CAAN;AACD,CAFD;;AAgBAlC,QAAQ,CAAC4G,cAAT,GAA0B,gBAAgB5E,IAAI,GAAG,EAAvB,EAA2B;AACnD,QAAM;AACJ0E,IAAAA,GADI;AAEJG,IAAAA,OAAO,EAAEC,GAFL;AAGJC,IAAAA;AAHI,MAIF/E,IAJJ;AAKA,SAAO,MAAM,KAAKzB,GAAL,CAASoG,QAAT,CAAkBD,GAAlB,EAAuBI,GAAvB,EAA4B;AAAEC,IAAAA;AAAF,GAA5B,CAAb;AACD,CAPD;;AASA/G,QAAQ,CAACgH,iBAAT,GAA6B,kBAAkB;AAC7C,SAAO,MAAM,KAAK7G,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,mCAAlC,EAAuE,MAAvE,EAA+E,EAA/E,CAAb;AACD,CAFD;;AAIAL,QAAQ,CAACiH,cAAT,GAA0B,gBAAgBC,QAAhB,EAA0B;AAKlD,MAAIC,kBAAkB,GAAG,EAAzB;AACA,MAAIC,cAAc,GAAG,EAArB;;AACA,OAAK,IAAI,CAACC,OAAD,EAAU5F,KAAV,CAAT,IAA6BC,gBAAEC,OAAF,CAAUuF,QAAV,CAA7B,EAAkD;AAChD,QAAII,8CAA4BC,QAA5B,CAAqCF,OAArC,CAAJ,EAAmD;AACjDF,MAAAA,kBAAkB,CAACE,OAAD,CAAlB,GAA8B5F,KAA9B;AACD,KAFD,MAEO;AACL2F,MAAAA,cAAc,CAACC,OAAD,CAAd,GAA0B5F,KAA1B;AACD;AACF;;AACD,MAAI,CAACC,gBAAE8F,OAAF,CAAUL,kBAAV,CAAL,EAAoC;AAClCrG,oBAAIM,IAAJ,CAAU,4DAAD,GACC,GAAES,IAAI,CAACC,SAAL,CAAeJ,gBAAE0E,IAAF,CAAOe,kBAAP,CAAf,CAA2C,QAD9C,GAEC,0DAFD,GAGC,gCAHV;;AAIA,UAAM,KAAKD,QAAL,CAAcO,MAAd,CAAqBN,kBAArB,CAAN;AACD;;AACD,MAAI,CAACzF,gBAAE8F,OAAF,CAAUJ,cAAV,CAAL,EAAgC;AAC9BtG,oBAAIM,IAAJ,CAAS,mEACAS,IAAI,CAACC,SAAL,CAAeJ,gBAAE0E,IAAF,CAAOgB,cAAP,CAAf,CADT;;AAEA,UAAM,KAAKjH,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,kBAAlC,EAAsD,MAAtD,EACJ;AAAC6G,MAAAA,QAAQ,EAAEE;AAAX,KADI,CAAN;AAED;AACF,CA3BD;;AA6BApH,QAAQ,CAAC0H,WAAT,GAAuB,kBAAkB;AAEvC,QAAMP,kBAAkB,GAAG,KAAKD,QAAL,CAAcQ,WAAd,EAA3B;AACA,QAAMN,cAAc,GAAG,MAAM,KAAKjH,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,kBAAlC,EAAsD,KAAtD,CAA7B;AACA,SAAO,EAAC,GAAG8G,kBAAJ;AAAwB,OAAGC;AAA3B,GAAP;AACD,CALD;;AAYAnH,OAAO,CAAC0H,uBAAR,GAAkC,gBAAgBC,OAAhB,EAAyB;AACzD,QAAMA,OAAO,EAAb;AACD,CAFD;;AAKA3H,OAAO,CAAC4H,wBAAR,GAAmC,YAAY;AAC7C,OAAKC,YAAL,GAAoB,IAApB;AACA,OAAKC,WAAL,GAAmB,KAAK5H,YAAL,CAAkB4H,WAAlB,CAA8BC,IAA9B,CAAmC,KAAK7H,YAAxC,CAAnB;AACA,OAAK8H,YAAL,GAAoB,KAAK9H,YAAL,CAAkB8H,YAAlB,CAA+BD,IAA/B,CAAoC,KAAK7H,YAAzC,CAApB;AACA,OAAK+H,cAAL,GAAsB,IAAtB;AACD,CALD;;AAWAlI,QAAQ,CAACmI,mBAAT,GAA+B,kBAAkB;AAC/C,SAAO,MAAM,KAAKhI,YAAL,CAAkBC,OAAlB,CAA0BC,OAA1B,CAAkC,qBAAlC,EAAyD,KAAzD,CAAb;AACD,CAFD;;AAkBAL,QAAQ,CAACoI,UAAT,GAAsB,eAAeA,UAAf,CAA2BpG,IAAI,GAAG,EAAlC,EAAsC;AAC1D,QAAM;AACJqG,IAAAA;AADI,MAEFrG,IAFJ;;AAGA,MAAIN,gBAAE4G,WAAF,CAAcD,IAAd,CAAJ,EAAyB;AACvBvH,oBAAIqB,aAAJ,CAAmB,kCAAnB;AACD;;AACD,SAAO,MAAM,KAAK5B,GAAL,CAASgI,WAAT,CAAqBF,IAArB,CAAb;AACD,CARD;;AAyCArI,QAAQ,CAACwI,yBAAT,GAAqC,gBAAgBxG,IAAI,GAAG,EAAvB,EAA2B;AAC9D,MAAI,CAACN,gBAAE+G,OAAF,CAAUzG,IAAI,CAAC0G,IAAf,CAAD,IAAyBhH,gBAAE8F,OAAF,CAAUxF,IAAI,CAAC0G,IAAf,CAA7B,EAAmD;AACjD,UAAM,IAAIxC,yBAAOyC,oBAAX,CAAgC,8BAAhC,CAAN;AACD;;AACD,QAAMD,IAAI,GAAG,MAAME,kBAAEC,GAAF,CAAM7G,IAAI,CAAC0G,IAAL,CACtBI,GADsB,CACjB7G,GAAD,IAAS,KAAKhC,OAAL,CAAa8I,YAAb,CAA0B9G,GAA1B,EAA+B,CAAC+G,yBAAD,CAA/B,CADS,CAAN,CAAnB;AAEA,QAAM,KAAKzI,GAAL,CAASwF,mBAAT,CAA6B2C,IAA7B,EAAmC1G,IAAI,CAACiH,OAAxC,CAAN;AACD,CAPD;;AASAC,MAAM,CAACC,MAAP,CAAcpJ,UAAd,EAA0BC,QAA1B,EAAoCC,OAApC;eAEeF,U","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport B from 'bluebird';\nimport { errors, BASEDRIVER_HANDLED_SETTINGS } from 'appium-base-driver';\nimport { fs, tempDir } from 'appium-support';\nimport { APK_EXTENSION } from '../extensions';\n\nlet extensions = {},\n    commands = {},\n    helpers = {};\n\ncommands.getPageSource = async function () {\n  return await this.uiautomator2.jwproxy.command('/source', 'GET', {});\n};\n\ncommands.getClipboard = async function () {\n  return (await this.adb.getApiLevel() < 29)\n    ? (await this.uiautomator2.jwproxy.command('/appium/device/get_clipboard', 'POST', {}))\n    : (await this.adb.getClipboard());\n};\n\n// Need to override this for correct unicode support\ncommands.doSendKeys = async function (params) {\n  await this.uiautomator2.jwproxy.command('/keys', 'POST', params);\n};\n\n// uiautomator2 doesn't support metastate for keyevents\ncommands.keyevent = async function (keycode, metastate) {\n  log.debug(`Ignoring metastate ${metastate}`);\n  await this.adb.keyevent(keycode);\n};\n\n// Use ADB since we don't have UiAutomator\ncommands.back = async function () {\n  await this.adb.keyevent(4);\n};\n\ncommands.getStrings = async function (language) {\n  if (!language) {\n    language = await this.adb.getDeviceLanguage();\n    log.info(`No language specified, returning strings for: ${language}`);\n  }\n\n  // Clients require the resulting mapping to have both keys\n  // and values of type string\n  const preprocessStringsMap = function (mapping) {\n    const result = {};\n    for (const [key, value] of _.toPairs(mapping)) {\n      result[key] = _.isString(value) ? value : JSON.stringify(value);\n    }\n    return result;\n  };\n\n  if (this.apkStrings[language]) {\n    // Return cached strings\n    return preprocessStringsMap(this.apkStrings[language]);\n  }\n\n  if (!this.opts.app && !this.opts.appPackage) {\n    log.errorAndThrow(\"One of 'app' or 'appPackage' capabilities should must be specified\");\n  }\n\n  let app = this.opts.app;\n  const tmpRoot = await tempDir.openDir();\n  try {\n    if (!app) {\n      try {\n        app = await this.adb.pullApk(this.opts.appPackage, tmpRoot);\n      } catch (err) {\n        log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);\n      }\n    }\n\n    if (!await fs.exists(app)) {\n      log.errorAndThrow(`The app at '${app}' does not exist`);\n    }\n\n    try {\n      const {apkStrings} = await this.adb.extractStringsFromApk(app, language, tmpRoot);\n      this.apkStrings[language] = apkStrings;\n      return preprocessStringsMap(apkStrings);\n    } catch (err) {\n      log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);\n    }\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n};\n\ncommands.getDisplayDensity = async function getDisplayDensity () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/display_density', 'GET', {});\n};\n\n// memoized in constructor\ncommands.getWindowSize = async function () {\n  return await this.uiautomator2.jwproxy.command('/window/current/size', 'GET', {});\n};\n\n// For W3C\ncommands.getWindowRect = async function () {\n  const {width, height} = await this.getWindowSize();\n  return {\n    width,\n    height,\n    x: 0,\n    y: 0,\n  };\n};\n\nextensions.executeMobile = async function (mobileCommand, opts = {}) {\n  const mobileCommandsMapping = {\n    shell: 'mobileShell',\n\n    execEmuConsoleCommand: 'mobileExecEmuConsoleCommand',\n\n    dragGesture: 'mobileDragGesture',\n    flingGesture: 'mobileFlingGesture',\n    doubleClickGesture: 'mobileDoubleClickGesture',\n    clickGesture: 'mobileClickGesture',\n    longClickGesture: 'mobileLongClickGesture',\n    pinchCloseGesture: 'mobilePinchCloseGesture',\n    pinchOpenGesture: 'mobilePinchOpenGesture',\n    swipeGesture: 'mobileSwipeGesture',\n    scrollGesture: 'mobileScrollGesture',\n    scrollBackTo: 'mobileScrollBackTo',\n    scroll: 'mobileScroll',\n    viewportScreenshot: 'mobileViewportScreenshot',\n    viewportRect: 'mobileViewPortRect',\n\n    deepLink: 'mobileDeepLink',\n\n    startLogsBroadcast: 'mobileStartLogsBroadcast',\n    stopLogsBroadcast: 'mobileStopLogsBroadcast',\n\n    acceptAlert: 'mobileAcceptAlert',\n    dismissAlert: 'mobileDismissAlert',\n\n    batteryInfo: 'mobileGetBatteryInfo',\n\n    deviceInfo: 'mobileGetDeviceInfo',\n\n    getDeviceTime: 'mobileGetDeviceTime',\n\n    changePermissions: 'mobileChangePermissions',\n    getPermissions: 'mobileGetPermissions',\n\n    performEditorAction: 'mobilePerformEditorAction',\n\n    startScreenStreaming: 'mobileStartScreenStreaming',\n    stopScreenStreaming: 'mobileStopScreenStreaming',\n\n    getNotifications: 'mobileGetNotifications',\n\n    listSms: 'mobileListSms',\n\n    type: 'mobileType',\n    sensorSet: 'sensorSet',\n\n    deleteFile: 'mobileDeleteFile',\n\n    clearApp: 'mobileClearApp',\n\n    startActivity: 'mobileStartActivity',\n    startService: 'mobileStartService',\n    stopService: 'mobileStopService',\n    broadcast: 'mobileBroadcast',\n\n    getContexts: 'mobileGetContexts',\n\n    installMultipleApks: 'mobileInstallMultipleApks',\n\n    unlock: 'mobileUnlock',\n  };\n\n  if (!_.has(mobileCommandsMapping, mobileCommand)) {\n    throw new errors.UnknownCommandError(`Unknown mobile command \"${mobileCommand}\". ` +\n      `Only ${_.keys(mobileCommandsMapping)} commands are supported.`);\n  }\n  return await this[mobileCommandsMapping[mobileCommand]](opts);\n};\n\ncommands.mobileViewportScreenshot = async function () {\n  return await this.getViewportScreenshot();\n};\n\n/**\n * @typedef {object} Rectangle\n * @property {number} left - The left coordinate of the Rectangle.\n * @property {number} top - The top coordinate of the Rectangle.\n * @property {number} width - The width of Rectangle.\n * @property {number} height - The height of Rectangle.\n */\n\n/**\n * Returns the viewport coordinates.\n * @returns {Rectangle} The viewport coordinates.\n */\ncommands.mobileViewPortRect = async function mobileViewPortRect () {\n  return await this.getViewPortRect();\n};\n\ncommands.setUrl = async function (url) {\n  await this.adb.startUri(url, this.opts.appPackage);\n};\n\n/**\n * @typedef {object} DeepLinkOpts\n * @property {!string} url - The name of URL to start.\n * @property {!string} package - The name of the package to start the URI with.\n * @property {?boolean} waitForLaunch [true] - if `false` then adb won't wait\n * for the started activity to return the control\n */\n\n/**\n * Start URL that take users directly to specific content in the app\n * @param {DeepLinkOpts} opts\n */\ncommands.mobileDeepLink = async function (opts = {}) {\n  const {\n    url,\n    package: pkg,\n    waitForLaunch,\n  } = opts;\n  return await this.adb.startUri(url, pkg, { waitForLaunch });\n};\n\ncommands.openNotifications = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/open_notifications', 'POST', {});\n};\n\ncommands.updateSettings = async function (settings) {\n  // we have some settings that are set on the settings object in the driver\n  // only, for example image finding settings. The uiauto2 server does not know\n  // what to do with them, so just set them on this driver's settings instance,\n  // and don't forward them to the server\n  let driverOnlySettings = {};\n  let serverSettings = {};\n  for (let [setting, value] of _.toPairs(settings)) {\n    if (BASEDRIVER_HANDLED_SETTINGS.includes(setting)) {\n      driverOnlySettings[setting] = value;\n    } else {\n      serverSettings[setting] = value;\n    }\n  }\n  if (!_.isEmpty(driverOnlySettings)) {\n    log.info(`Found some settings designed to be handled by BaseDriver: ` +\n             `${JSON.stringify(_.keys(driverOnlySettings))}. Not ` +\n             `sending these on to the UiAutomator2 server and instead ` +\n             `setting directly on the driver`);\n    await this.settings.update(driverOnlySettings);\n  }\n  if (!_.isEmpty(serverSettings)) {\n    log.info('Forwarding the following settings to the UiAutomator2 server: ' +\n             JSON.stringify(_.keys(serverSettings)));\n    await this.uiautomator2.jwproxy.command('/appium/settings', 'POST',\n      {settings: serverSettings});\n  }\n};\n\ncommands.getSettings = async function () {\n  // as above, we might have some driver-only settings to return as well\n  const driverOnlySettings = this.settings.getSettings();\n  const serverSettings = await this.uiautomator2.jwproxy.command('/appium/settings', 'GET');\n  return {...driverOnlySettings, ...serverSettings};\n};\n\n/**\n * Overriding appium-android-driver's wrapBootstrapDisconnect,\n * unlike in appium-android-driver avoiding adb restarting as it intern\n * kills UiAutomator2 server running in the device.\n **/\nhelpers.wrapBootstrapDisconnect = async function (wrapped) {\n  await wrapped();\n};\n\n// Stop proxying to any Chromedriver and redirect to uiautomator2\nhelpers.suspendChromedriverProxy = function () {\n  this.chromedriver = null;\n  this.proxyReqRes = this.uiautomator2.proxyReqRes.bind(this.uiautomator2);\n  this.proxyCommand = this.uiautomator2.proxyCommand.bind(this.uiautomator2);\n  this.jwpProxyActive = true;\n};\n\n/**\n * The list of available info entries can be found at\n * https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/GetDeviceInfo.java\n */\ncommands.mobileGetDeviceInfo = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/info', 'GET');\n};\n\n/**\n * @typedef {Object} TypingOptions\n * @property {!string|number|boolean} text - The text to type\n */\n\n/**\n * Types the given Unicode string.\n * It is expected that the focus is already put\n * to the destination input field before this method is called.\n *\n * @param {TypingOptions} opts\n * @returns {boolean} `true` if the input text has been successfully sent to adb\n * @throws {Error} if `text` property has not been provided\n */\ncommands.mobileType = async function mobileType (opts = {}) {\n  const {\n    text,\n  } = opts;\n  if (_.isUndefined(text)) {\n    log.errorAndThrow(`The 'text' argument is mandatory`);\n  }\n  return await this.adb.typeUnicode(text);\n};\n\n\n/**\n * @typedef {Object} InstallOptions\n * @property {boolean} allowTestPackages [false] - Set to true in order to allow test\n *                                                 packages installation.\n * @property {boolean} useSdcard [false] - Set to true to install the app on sdcard\n *                                         instead of the device memory.\n * @property {boolean} grantPermissions [false] - Set to true in order to grant all the\n *                                                permissions requested in the application's manifest\n *                                                automatically after the installation is completed\n *                                                under Android 6+.\n * @property {boolean} replace [true] - Set it to false if you don't want\n *                                      the application to be upgraded/reinstalled\n *                                      if it is already present on the device.\n * @property {boolean} partialInstall [false] - Install apks partially. It is used for 'install-multiple'.\n *                                             https://android.stackexchange.com/questions/111064/what-is-a-partial-application-install-via-adb\n */\n\n/**\n * @typedef {Object} InstallMultipleApksOptions\n * @property {Array<string>} apks - The list of APKs to install. Each APK should be a path to a apk\n *                                  or downloadable URL as HTTP/HTTPS.\n * @property {InstallOptions} options\n */\n\n/**\n * Install multiple APKs with `install-multiple` option.\n *\n * @param {InstallMultipleApksOptions} opts\n * @throws {Error} if an error occured while installing the given APKs.\n */\ncommands.mobileInstallMultipleApks = async function (opts = {}) {\n  if (!_.isArray(opts.apks) || _.isEmpty(opts.apks)) {\n    throw new errors.InvalidArgumentError('No apks are given to install');\n  }\n  const apks = await B.all(opts.apks\n    .map((app) => this.helpers.configureApp(app, [APK_EXTENSION])));\n  await this.adb.installMultipleApks(apks, opts.options);\n};\n\nObject.assign(extensions, commands, helpers);\n\nexport default extensions;\n"],"file":"lib/commands/general.js","sourceRoot":"../../.."}
282
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/general.js"],"names":["extensions","commands","helpers","getPageSource","uiautomator2","jwproxy","command","getClipboard","adb","getApiLevel","doSendKeys","params","keyevent","keycode","metastate","log","debug","back","getStrings","language","getDeviceLanguage","info","preprocessStringsMap","mapping","result","key","value","_","toPairs","isString","JSON","stringify","apkStrings","opts","app","appPackage","errorAndThrow","tmpRoot","tempDir","openDir","pullApk","err","message","fs","exists","extractStringsFromApk","rimraf","getDisplayDensity","getWindowSize","getWindowRect","width","height","x","y","executeMobile","mobileCommand","mobileCommandsMapping","shell","execEmuConsoleCommand","dragGesture","flingGesture","doubleClickGesture","clickGesture","longClickGesture","pinchCloseGesture","pinchOpenGesture","swipeGesture","scrollGesture","scrollBackTo","scroll","viewportScreenshot","viewportRect","deepLink","startLogsBroadcast","stopLogsBroadcast","acceptAlert","dismissAlert","batteryInfo","deviceInfo","getDeviceTime","changePermissions","getPermissions","performEditorAction","startScreenStreaming","stopScreenStreaming","getNotifications","listSms","type","sensorSet","deleteFile","clearApp","startActivity","startService","stopService","broadcast","getContexts","installMultipleApks","unlock","refreshGpsCache","has","errors","UnknownCommandError","keys","mobileViewportScreenshot","getViewportScreenshot","mobileViewPortRect","getViewPortRect","setUrl","url","startUri","mobileDeepLink","package","pkg","waitForLaunch","openNotifications","updateSettings","settings","driverOnlySettings","serverSettings","setting","BASEDRIVER_HANDLED_SETTINGS","includes","isEmpty","update","getSettings","wrapBootstrapDisconnect","wrapped","suspendChromedriverProxy","chromedriver","proxyReqRes","bind","proxyCommand","jwpProxyActive","mobileGetDeviceInfo","mobileType","text","isUndefined","typeUnicode","mobileInstallMultipleApks","isArray","apks","InvalidArgumentError","B","all","map","configureApp","APK_EXTENSION","options","Object","assign"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport B from 'bluebird';\nimport { errors, BASEDRIVER_HANDLED_SETTINGS } from 'appium-base-driver';\nimport { fs, tempDir } from 'appium-support';\nimport { APK_EXTENSION } from '../extensions';\n\nlet extensions = {},\n    commands = {},\n    helpers = {};\n\ncommands.getPageSource = async function () {\n  return await this.uiautomator2.jwproxy.command('/source', 'GET', {});\n};\n\ncommands.getClipboard = async function () {\n  return (await this.adb.getApiLevel() < 29)\n    ? (await this.uiautomator2.jwproxy.command('/appium/device/get_clipboard', 'POST', {}))\n    : (await this.adb.getClipboard());\n};\n\n// Need to override this for correct unicode support\ncommands.doSendKeys = async function (params) {\n  await this.uiautomator2.jwproxy.command('/keys', 'POST', params);\n};\n\n// uiautomator2 doesn't support metastate for keyevents\ncommands.keyevent = async function (keycode, metastate) {\n  log.debug(`Ignoring metastate ${metastate}`);\n  await this.adb.keyevent(keycode);\n};\n\n// Use ADB since we don't have UiAutomator\ncommands.back = async function () {\n  await this.adb.keyevent(4);\n};\n\ncommands.getStrings = async function (language) {\n  if (!language) {\n    language = await this.adb.getDeviceLanguage();\n    log.info(`No language specified, returning strings for: ${language}`);\n  }\n\n  // Clients require the resulting mapping to have both keys\n  // and values of type string\n  const preprocessStringsMap = function (mapping) {\n    const result = {};\n    for (const [key, value] of _.toPairs(mapping)) {\n      result[key] = _.isString(value) ? value : JSON.stringify(value);\n    }\n    return result;\n  };\n\n  if (this.apkStrings[language]) {\n    // Return cached strings\n    return preprocessStringsMap(this.apkStrings[language]);\n  }\n\n  if (!this.opts.app && !this.opts.appPackage) {\n    log.errorAndThrow(\"One of 'app' or 'appPackage' capabilities should must be specified\");\n  }\n\n  let app = this.opts.app;\n  const tmpRoot = await tempDir.openDir();\n  try {\n    if (!app) {\n      try {\n        app = await this.adb.pullApk(this.opts.appPackage, tmpRoot);\n      } catch (err) {\n        log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);\n      }\n    }\n\n    if (!await fs.exists(app)) {\n      log.errorAndThrow(`The app at '${app}' does not exist`);\n    }\n\n    try {\n      const {apkStrings} = await this.adb.extractStringsFromApk(app, language, tmpRoot);\n      this.apkStrings[language] = apkStrings;\n      return preprocessStringsMap(apkStrings);\n    } catch (err) {\n      log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);\n    }\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n};\n\ncommands.getDisplayDensity = async function getDisplayDensity () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/display_density', 'GET', {});\n};\n\n// memoized in constructor\ncommands.getWindowSize = async function () {\n  return await this.uiautomator2.jwproxy.command('/window/current/size', 'GET', {});\n};\n\n// For W3C\ncommands.getWindowRect = async function () {\n  const {width, height} = await this.getWindowSize();\n  return {\n    width,\n    height,\n    x: 0,\n    y: 0,\n  };\n};\n\nextensions.executeMobile = async function (mobileCommand, opts = {}) {\n  const mobileCommandsMapping = {\n    shell: 'mobileShell',\n\n    execEmuConsoleCommand: 'mobileExecEmuConsoleCommand',\n\n    dragGesture: 'mobileDragGesture',\n    flingGesture: 'mobileFlingGesture',\n    doubleClickGesture: 'mobileDoubleClickGesture',\n    clickGesture: 'mobileClickGesture',\n    longClickGesture: 'mobileLongClickGesture',\n    pinchCloseGesture: 'mobilePinchCloseGesture',\n    pinchOpenGesture: 'mobilePinchOpenGesture',\n    swipeGesture: 'mobileSwipeGesture',\n    scrollGesture: 'mobileScrollGesture',\n    scrollBackTo: 'mobileScrollBackTo',\n    scroll: 'mobileScroll',\n    viewportScreenshot: 'mobileViewportScreenshot',\n    viewportRect: 'mobileViewPortRect',\n\n    deepLink: 'mobileDeepLink',\n\n    startLogsBroadcast: 'mobileStartLogsBroadcast',\n    stopLogsBroadcast: 'mobileStopLogsBroadcast',\n\n    acceptAlert: 'mobileAcceptAlert',\n    dismissAlert: 'mobileDismissAlert',\n\n    batteryInfo: 'mobileGetBatteryInfo',\n\n    deviceInfo: 'mobileGetDeviceInfo',\n\n    getDeviceTime: 'mobileGetDeviceTime',\n\n    changePermissions: 'mobileChangePermissions',\n    getPermissions: 'mobileGetPermissions',\n\n    performEditorAction: 'mobilePerformEditorAction',\n\n    startScreenStreaming: 'mobileStartScreenStreaming',\n    stopScreenStreaming: 'mobileStopScreenStreaming',\n\n    getNotifications: 'mobileGetNotifications',\n\n    listSms: 'mobileListSms',\n\n    type: 'mobileType',\n    sensorSet: 'sensorSet',\n\n    deleteFile: 'mobileDeleteFile',\n\n    clearApp: 'mobileClearApp',\n\n    startActivity: 'mobileStartActivity',\n    startService: 'mobileStartService',\n    stopService: 'mobileStopService',\n    broadcast: 'mobileBroadcast',\n\n    getContexts: 'mobileGetContexts',\n\n    installMultipleApks: 'mobileInstallMultipleApks',\n\n    unlock: 'mobileUnlock',\n\n    refreshGpsCache: 'mobileRefreshGpsCache',\n  };\n\n  if (!_.has(mobileCommandsMapping, mobileCommand)) {\n    throw new errors.UnknownCommandError(`Unknown mobile command \"${mobileCommand}\". ` +\n      `Only ${_.keys(mobileCommandsMapping)} commands are supported.`);\n  }\n  return await this[mobileCommandsMapping[mobileCommand]](opts);\n};\n\ncommands.mobileViewportScreenshot = async function () {\n  return await this.getViewportScreenshot();\n};\n\n/**\n * @typedef {object} Rectangle\n * @property {number} left - The left coordinate of the Rectangle.\n * @property {number} top - The top coordinate of the Rectangle.\n * @property {number} width - The width of Rectangle.\n * @property {number} height - The height of Rectangle.\n */\n\n/**\n * Returns the viewport coordinates.\n * @returns {Rectangle} The viewport coordinates.\n */\ncommands.mobileViewPortRect = async function mobileViewPortRect () {\n  return await this.getViewPortRect();\n};\n\ncommands.setUrl = async function (url) {\n  await this.adb.startUri(url, this.opts.appPackage);\n};\n\n/**\n * @typedef {object} DeepLinkOpts\n * @property {!string} url - The name of URL to start.\n * @property {!string} package - The name of the package to start the URI with.\n * @property {?boolean} waitForLaunch [true] - if `false` then adb won't wait\n * for the started activity to return the control\n */\n\n/**\n * Start URL that take users directly to specific content in the app\n * @param {DeepLinkOpts} opts\n */\ncommands.mobileDeepLink = async function (opts = {}) {\n  const {\n    url,\n    package: pkg,\n    waitForLaunch,\n  } = opts;\n  return await this.adb.startUri(url, pkg, { waitForLaunch });\n};\n\ncommands.openNotifications = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/open_notifications', 'POST', {});\n};\n\ncommands.updateSettings = async function (settings) {\n  // we have some settings that are set on the settings object in the driver\n  // only, for example image finding settings. The uiauto2 server does not know\n  // what to do with them, so just set them on this driver's settings instance,\n  // and don't forward them to the server\n  let driverOnlySettings = {};\n  let serverSettings = {};\n  for (let [setting, value] of _.toPairs(settings)) {\n    if (BASEDRIVER_HANDLED_SETTINGS.includes(setting)) {\n      driverOnlySettings[setting] = value;\n    } else {\n      serverSettings[setting] = value;\n    }\n  }\n  if (!_.isEmpty(driverOnlySettings)) {\n    log.info(`Found some settings designed to be handled by BaseDriver: ` +\n             `${JSON.stringify(_.keys(driverOnlySettings))}. Not ` +\n             `sending these on to the UiAutomator2 server and instead ` +\n             `setting directly on the driver`);\n    await this.settings.update(driverOnlySettings);\n  }\n  if (!_.isEmpty(serverSettings)) {\n    log.info('Forwarding the following settings to the UiAutomator2 server: ' +\n             JSON.stringify(_.keys(serverSettings)));\n    await this.uiautomator2.jwproxy.command('/appium/settings', 'POST',\n      {settings: serverSettings});\n  }\n};\n\ncommands.getSettings = async function () {\n  // as above, we might have some driver-only settings to return as well\n  const driverOnlySettings = this.settings.getSettings();\n  const serverSettings = await this.uiautomator2.jwproxy.command('/appium/settings', 'GET');\n  return {...driverOnlySettings, ...serverSettings};\n};\n\n/**\n * Overriding appium-android-driver's wrapBootstrapDisconnect,\n * unlike in appium-android-driver avoiding adb restarting as it intern\n * kills UiAutomator2 server running in the device.\n **/\nhelpers.wrapBootstrapDisconnect = async function (wrapped) {\n  await wrapped();\n};\n\n// Stop proxying to any Chromedriver and redirect to uiautomator2\nhelpers.suspendChromedriverProxy = function () {\n  this.chromedriver = null;\n  this.proxyReqRes = this.uiautomator2.proxyReqRes.bind(this.uiautomator2);\n  this.proxyCommand = this.uiautomator2.proxyCommand.bind(this.uiautomator2);\n  this.jwpProxyActive = true;\n};\n\n/**\n * The list of available info entries can be found at\n * https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/GetDeviceInfo.java\n */\ncommands.mobileGetDeviceInfo = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/info', 'GET');\n};\n\n/**\n * @typedef {Object} TypingOptions\n * @property {!string|number|boolean} text - The text to type\n */\n\n/**\n * Types the given Unicode string.\n * It is expected that the focus is already put\n * to the destination input field before this method is called.\n *\n * @param {TypingOptions} opts\n * @returns {boolean} `true` if the input text has been successfully sent to adb\n * @throws {Error} if `text` property has not been provided\n */\ncommands.mobileType = async function mobileType (opts = {}) {\n  const {\n    text,\n  } = opts;\n  if (_.isUndefined(text)) {\n    log.errorAndThrow(`The 'text' argument is mandatory`);\n  }\n  return await this.adb.typeUnicode(text);\n};\n\n\n/**\n * @typedef {Object} InstallOptions\n * @property {boolean} allowTestPackages [false] - Set to true in order to allow test\n *                                                 packages installation.\n * @property {boolean} useSdcard [false] - Set to true to install the app on sdcard\n *                                         instead of the device memory.\n * @property {boolean} grantPermissions [false] - Set to true in order to grant all the\n *                                                permissions requested in the application's manifest\n *                                                automatically after the installation is completed\n *                                                under Android 6+.\n * @property {boolean} replace [true] - Set it to false if you don't want\n *                                      the application to be upgraded/reinstalled\n *                                      if it is already present on the device.\n * @property {boolean} partialInstall [false] - Install apks partially. It is used for 'install-multiple'.\n *                                             https://android.stackexchange.com/questions/111064/what-is-a-partial-application-install-via-adb\n */\n\n/**\n * @typedef {Object} InstallMultipleApksOptions\n * @property {Array<string>} apks - The list of APKs to install. Each APK should be a path to a apk\n *                                  or downloadable URL as HTTP/HTTPS.\n * @property {InstallOptions} options\n */\n\n/**\n * Install multiple APKs with `install-multiple` option.\n *\n * @param {InstallMultipleApksOptions} opts\n * @throws {Error} if an error occured while installing the given APKs.\n */\ncommands.mobileInstallMultipleApks = async function (opts = {}) {\n  if (!_.isArray(opts.apks) || _.isEmpty(opts.apks)) {\n    throw new errors.InvalidArgumentError('No apks are given to install');\n  }\n  const apks = await B.all(opts.apks\n    .map((app) => this.helpers.configureApp(app, [APK_EXTENSION])));\n  await this.adb.installMultipleApks(apks, opts.options);\n};\n\nObject.assign(extensions, commands, helpers);\n\nexport default extensions;\n"],"file":"lib/commands/general.js","sourceRoot":"../../.."}
@@ -13,7 +13,6 @@ var _lodash = require("lodash");
13
13
 
14
14
  var _appiumBaseDriver = require("appium-base-driver");
15
15
 
16
- const CssConverter = {};
17
16
  const parser = new _cssSelectorParser.CssSelectorParser();
18
17
  parser.registerSelectorPseudos('has');
19
18
  parser.registerNestingOperators('>', '+', '~');
@@ -69,165 +68,173 @@ function getWordMatcherRegex(word) {
69
68
  return `\\b(\\w*${(0, _lodash.escapeRegExp)(word)}\\w*)\\b`;
70
69
  }
71
70
 
72
- function formatIdLocator(locator) {
73
- return ID_LOCATOR_PATTERN.test(locator) ? locator : `android:id/${locator}`;
74
- }
71
+ class CssConverter {
72
+ constructor(selector, pkg) {
73
+ this.selector = selector;
74
+ this.pkg = pkg;
75
+ }
75
76
 
76
- function parseAttr(cssAttr) {
77
- if (cssAttr.valueType && cssAttr.valueType !== 'string') {
78
- throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` + `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);
77
+ formatIdLocator(locator) {
78
+ return ID_LOCATOR_PATTERN.test(locator) ? locator : `${this.pkg || 'android'}:id/${locator}`;
79
79
  }
80
80
 
81
- const attrName = assertGetAttrName(cssAttr);
82
- const methodName = toSnakeCase(attrName);
81
+ parseAttr(cssAttr) {
82
+ if (cssAttr.valueType && cssAttr.valueType !== 'string') {
83
+ throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` + `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);
84
+ }
83
85
 
84
- if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {
85
- throw new Error(`'${attrName}' is not supported. Supported attributes are ` + `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);
86
- }
86
+ const attrName = assertGetAttrName(cssAttr);
87
+ const methodName = toSnakeCase(attrName);
87
88
 
88
- if (BOOLEAN_ATTRS.includes(attrName)) {
89
- return `.${methodName}(${assertGetBool(cssAttr)})`;
90
- }
89
+ if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {
90
+ throw new Error(`'${attrName}' is not supported. Supported attributes are ` + `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);
91
+ }
91
92
 
92
- let value = cssAttr.value || '';
93
+ if (BOOLEAN_ATTRS.includes(attrName)) {
94
+ return `.${methodName}(${assertGetBool(cssAttr)})`;
95
+ }
93
96
 
94
- if (attrName === RESOURCE_ID) {
95
- value = formatIdLocator(value);
96
- }
97
+ let value = cssAttr.value || '';
97
98
 
98
- if (value === '') {
99
- return `.${methodName}Matches("")`;
100
- }
99
+ if (attrName === RESOURCE_ID) {
100
+ value = this.formatIdLocator(value);
101
+ }
101
102
 
102
- switch (cssAttr.operator) {
103
- case '=':
104
- return `.${methodName}("${value}")`;
103
+ if (value === '') {
104
+ return `.${methodName}Matches("")`;
105
+ }
105
106
 
106
- case '*=':
107
- if (['description', 'text'].includes(attrName)) {
108
- return `.${methodName}Contains("${value}")`;
109
- }
107
+ switch (cssAttr.operator) {
108
+ case '=':
109
+ return `.${methodName}("${value}")`;
110
110
 
111
- return `.${methodName}Matches("${(0, _lodash.escapeRegExp)(value)}")`;
111
+ case '*=':
112
+ if (['description', 'text'].includes(attrName)) {
113
+ return `.${methodName}Contains("${value}")`;
114
+ }
112
115
 
113
- case '^=':
114
- if (['description', 'text'].includes(attrName)) {
115
- return `.${methodName}StartsWith("${value}")`;
116
- }
116
+ return `.${methodName}Matches("${(0, _lodash.escapeRegExp)(value)}")`;
117
117
 
118
- return `.${methodName}Matches("^${(0, _lodash.escapeRegExp)(value)}")`;
118
+ case '^=':
119
+ if (['description', 'text'].includes(attrName)) {
120
+ return `.${methodName}StartsWith("${value}")`;
121
+ }
119
122
 
120
- case '$=':
121
- return `.${methodName}Matches("${(0, _lodash.escapeRegExp)(value)}$")`;
123
+ return `.${methodName}Matches("^${(0, _lodash.escapeRegExp)(value)}")`;
122
124
 
123
- case '~=':
124
- return `.${methodName}Matches("${getWordMatcherRegex(value)}")`;
125
+ case '$=':
126
+ return `.${methodName}Matches("${(0, _lodash.escapeRegExp)(value)}$")`;
125
127
 
126
- default:
127
- throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` + ` '=', '*=', '^=', '$=' and '~=' are supported.`);
128
- }
129
- }
128
+ case '~=':
129
+ return `.${methodName}Matches("${getWordMatcherRegex(value)}")`;
130
130
 
131
- function parsePseudo(cssPseudo) {
132
- if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {
133
- throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` + `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);
131
+ default:
132
+ throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` + ` '=', '*=', '^=', '$=' and '~=' are supported.`);
133
+ }
134
134
  }
135
135
 
136
- const pseudoName = assertGetAttrName(cssPseudo);
136
+ parsePseudo(cssPseudo) {
137
+ if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {
138
+ throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` + `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);
139
+ }
137
140
 
138
- if (BOOLEAN_ATTRS.includes(pseudoName)) {
139
- return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;
140
- }
141
+ const pseudoName = assertGetAttrName(cssPseudo);
141
142
 
142
- if (NUMERIC_ATTRS.includes(pseudoName)) {
143
- return `.${pseudoName}(${cssPseudo.value})`;
143
+ if (BOOLEAN_ATTRS.includes(pseudoName)) {
144
+ return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;
145
+ }
146
+
147
+ if (NUMERIC_ATTRS.includes(pseudoName)) {
148
+ return `.${pseudoName}(${cssPseudo.value})`;
149
+ }
144
150
  }
145
- }
146
151
 
147
- function parseCssRule(cssRule) {
148
- const {
149
- nestingOperator
150
- } = cssRule;
152
+ parseCssRule(cssRule) {
153
+ const {
154
+ nestingOperator
155
+ } = cssRule;
151
156
 
152
- if (nestingOperator && nestingOperator !== ' ') {
153
- throw new Error(`'${nestingOperator}' is not a supported combinator. ` + `Only child combinator (>) and descendant combinator are supported.`);
154
- }
157
+ if (nestingOperator && nestingOperator !== ' ') {
158
+ throw new Error(`'${nestingOperator}' is not a supported combinator. ` + `Only child combinator (>) and descendant combinator are supported.`);
159
+ }
155
160
 
156
- let uiAutomatorSelector = 'new UiSelector()';
161
+ let uiAutomatorSelector = 'new UiSelector()';
157
162
 
158
- if (cssRule.tagName && cssRule.tagName !== '*') {
159
- let androidClass = [cssRule.tagName];
163
+ if (cssRule.tagName && cssRule.tagName !== '*') {
164
+ let androidClass = [cssRule.tagName];
160
165
 
161
- if (cssRule.classNames) {
162
- for (const cssClassNames of cssRule.classNames) {
163
- androidClass.push(cssClassNames);
166
+ if (cssRule.classNames) {
167
+ for (const cssClassNames of cssRule.classNames) {
168
+ androidClass.push(cssClassNames);
169
+ }
170
+
171
+ uiAutomatorSelector += `.className("${androidClass.join('.')}")`;
172
+ } else {
173
+ uiAutomatorSelector += `.classNameMatches("${cssRule.tagName}")`;
164
174
  }
175
+ } else if (cssRule.classNames) {
176
+ uiAutomatorSelector += `.classNameMatches("${cssRule.classNames.join('\\.')}")`;
177
+ }
165
178
 
166
- uiAutomatorSelector += `.className("${androidClass.join('.')}")`;
167
- } else {
168
- uiAutomatorSelector += `.classNameMatches("${cssRule.tagName}")`;
179
+ if (cssRule.id) {
180
+ uiAutomatorSelector += `.resourceId("${this.formatIdLocator(cssRule.id)}")`;
169
181
  }
170
- } else if (cssRule.classNames) {
171
- uiAutomatorSelector += `.classNameMatches("${cssRule.classNames.join('\\.')}")`;
172
- }
173
182
 
174
- if (cssRule.id) {
175
- uiAutomatorSelector += `.resourceId("${formatIdLocator(cssRule.id)}")`;
176
- }
183
+ if (cssRule.attrs) {
184
+ for (const attr of cssRule.attrs) {
185
+ uiAutomatorSelector += this.parseAttr(attr);
186
+ }
187
+ }
177
188
 
178
- if (cssRule.attrs) {
179
- for (const attr of cssRule.attrs) {
180
- uiAutomatorSelector += parseAttr(attr);
189
+ if (cssRule.pseudos) {
190
+ for (const pseudo of cssRule.pseudos) {
191
+ uiAutomatorSelector += this.parsePseudo(pseudo);
192
+ }
181
193
  }
182
- }
183
194
 
184
- if (cssRule.pseudos) {
185
- for (const pseudo of cssRule.pseudos) {
186
- uiAutomatorSelector += parsePseudo(pseudo);
195
+ if (cssRule.rule) {
196
+ uiAutomatorSelector += `.childSelector(${this.parseCssRule(cssRule.rule)})`;
187
197
  }
188
- }
189
198
 
190
- if (cssRule.rule) {
191
- uiAutomatorSelector += `.childSelector(${parseCssRule(cssRule.rule)})`;
199
+ return uiAutomatorSelector;
192
200
  }
193
201
 
194
- return uiAutomatorSelector;
195
- }
196
-
197
- function parseCssObject(css) {
198
- switch (css.type) {
199
- case 'rule':
200
- return parseCssRule(css);
202
+ parseCssObject(css) {
203
+ switch (css.type) {
204
+ case 'rule':
205
+ return this.parseCssRule(css);
201
206
 
202
- case 'ruleSet':
203
- return parseCssObject(css.rule);
207
+ case 'ruleSet':
208
+ return this.parseCssObject(css.rule);
204
209
 
205
- case 'selectors':
206
- return css.selectors.map(selector => parseCssObject(selector)).join('; ');
210
+ case 'selectors':
211
+ return css.selectors.map(selector => this.parseCssObject(selector)).join('; ');
207
212
 
208
- default:
209
- throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);
213
+ default:
214
+ throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);
215
+ }
210
216
  }
211
- }
212
217
 
213
- CssConverter.toUiAutomatorSelector = function toUiAutomatorSelector(cssSelector) {
214
- let cssObj;
218
+ toUiAutomatorSelector() {
219
+ let cssObj;
215
220
 
216
- try {
217
- cssObj = parser.parse(cssSelector);
218
- } catch (e) {
219
- throw new _appiumBaseDriver.errors.InvalidSelectorError(`Invalid CSS selector '${cssSelector}'. Reason: '${e}'`);
220
- }
221
+ try {
222
+ cssObj = parser.parse(this.selector);
223
+ } catch (e) {
224
+ throw new _appiumBaseDriver.errors.InvalidSelectorError(`Invalid CSS selector '${this.selector}'. Reason: '${e}'`);
225
+ }
221
226
 
222
- try {
223
- return parseCssObject(cssObj);
224
- } catch (e) {
225
- throw new _appiumBaseDriver.errors.InvalidSelectorError(`Unsupported CSS selector '${cssSelector}'. Reason: '${e}'`);
227
+ try {
228
+ return this.parseCssObject(cssObj);
229
+ } catch (e) {
230
+ throw new _appiumBaseDriver.errors.InvalidSelectorError(`Unsupported CSS selector '${this.selector}'. Reason: '${e}'`);
231
+ }
226
232
  }
227
- };
233
+
234
+ }
228
235
 
229
236
  var _default = CssConverter;
230
237
  exports.default = _default;require('source-map-support').install();
231
238
 
232
239
 
233
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/css-converter.js"],"names":["CssConverter","parser","CssSelectorParser","registerSelectorPseudos","registerNestingOperators","registerAttrEqualityMods","enableSubstitutes","RESOURCE_ID","ID_LOCATOR_PATTERN","BOOLEAN_ATTRS","NUMERIC_ATTRS","STR_ATTRS","ALL_ATTRS","ATTRIBUTE_ALIASES","toSnakeCase","str","tokens","split","map","charAt","toUpperCase","slice","toLowerCase","out","join","assertGetBool","css","val","value","includes","Error","name","assertGetAttrName","attrName","officialAttr","aliasAttrs","getWordMatcherRegex","word","formatIdLocator","locator","test","parseAttr","cssAttr","valueType","methodName","operator","parsePseudo","cssPseudo","pseudoName","parseCssRule","cssRule","nestingOperator","uiAutomatorSelector","tagName","androidClass","classNames","cssClassNames","push","id","attrs","attr","pseudos","pseudo","rule","parseCssObject","type","selectors","selector","toUiAutomatorSelector","cssSelector","cssObj","parse","e","errors","InvalidSelectorError"],"mappings":"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","sourcesContent":["import { CssSelectorParser } from 'css-selector-parser';\nimport { escapeRegExp } from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst CssConverter = {};\n\nconst parser = new CssSelectorParser();\nparser.registerSelectorPseudos('has');\nparser.registerNestingOperators('>', '+', '~');\nparser.registerAttrEqualityMods('^', '$', '*', '~');\nparser.enableSubstitutes();\n\nconst RESOURCE_ID = 'resource-id';\nconst ID_LOCATOR_PATTERN = /^[a-zA-Z_][a-zA-Z0-9._]*:id\\/[\\S]+$/;\n\nconst BOOLEAN_ATTRS = [\n  'checkable', 'checked', 'clickable', 'enabled', 'focusable',\n  'focused', 'long-clickable', 'scrollable', 'selected',\n];\n\nconst NUMERIC_ATTRS = [\n  'index', 'instance',\n];\n\nconst STR_ATTRS = [\n  'description', RESOURCE_ID, 'text', 'class-name', 'package-name'\n];\n\nconst ALL_ATTRS = [\n  ...BOOLEAN_ATTRS,\n  ...NUMERIC_ATTRS,\n  ...STR_ATTRS,\n];\n\nconst ATTRIBUTE_ALIASES = [\n  [RESOURCE_ID, ['id']],\n  ['description', [\n    'content-description', 'content-desc',\n    'desc', 'accessibility-id',\n  ]],\n  ['index', ['nth-child']],\n];\n\n/**\n * Convert hyphen separated word to snake case\n *\n * @param {string} str\n * @returns {string} The hyphen separated word translated to snake case\n */\nfunction toSnakeCase (str) {\n  if (!str) {\n    return '';\n  }\n  const tokens = str.split('-').map((str) => str.charAt(0).toUpperCase() + str.slice(1).toLowerCase());\n  const out = tokens.join('');\n  return out.charAt(0).toLowerCase() + out.slice(1);\n}\n\n/**\n * @typedef {Object} CssNameValueObject\n * @property {?name} name The name of the CSS object\n * @property {?string} value The value of the CSS object\n */\n\n/**\n * Get the boolean from a CSS object. If empty, return true. If not true/false/empty, throw exception\n *\n * @param {CssNameValueObject} css A CSS object that has 'name' and 'value'\n * @returns {string} Either 'true' or 'false'. If value is empty, return 'true'\n */\nfunction assertGetBool (css) {\n  const val = css.value?.toLowerCase() || 'true'; // an omitted boolean attribute means 'true' (e.g.: input[checked] means checked is true)\n  if (['true', 'false'].includes(val)) {\n    return val;\n  }\n  throw new Error(`'${css.name}' must be true, false or empty. Found '${css.value}'`);\n}\n\n/**\n * Get the canonical form of a CSS attribute name\n *\n * Converts to lowercase and if an attribute name is an alias for something else, return\n * what it is an alias for\n *\n * @param {Object} css CSS object\n * @returns {string} The canonical attribute name\n */\nfunction assertGetAttrName (css) {\n  const attrName = css.name.toLowerCase();\n\n  // Check if it's supported and if it is, return it\n  if (ALL_ATTRS.includes(attrName)) {\n    return attrName.toLowerCase();\n  }\n\n  // If attrName is an alias for something else, return that\n  for (const [officialAttr, aliasAttrs] of ATTRIBUTE_ALIASES) {\n    if (aliasAttrs.includes(attrName)) {\n      return officialAttr;\n    }\n  }\n  throw new Error(`'${attrName}' is not a valid attribute. ` +\n    `Supported attributes are '${ALL_ATTRS.join(', ')}'`);\n}\n\n/**\n * Get a regex that matches a whole word. For the ~= CSS attribute selector.\n *\n * @param {string} word\n * @returns {string} A regex \"word\" matcher\n */\nfunction getWordMatcherRegex (word) {\n  return `\\\\b(\\\\w*${escapeRegExp(word)}\\\\w*)\\\\b`;\n}\n\n/**\n * Add android:id/ to beginning of string if it's not there already\n *\n * @param {string} locator The initial locator\n * @returns {string} String with `android:id/` prepended (if it wasn't already)\n */\nfunction formatIdLocator (locator) {\n  return ID_LOCATOR_PATTERN.test(locator) ? locator : `android:id/${locator}`;\n}\n\n/**\n * @typedef {Object} CssAttr\n * @property {?string} valueType Type of attribute (must be string or empty)\n * @property {?string} value Value of the attribute\n * @property {?string} operator The operator between value and value type (=, *=, , ^=, $=)\n */\n\n/**\n * Convert a CSS attribute into a UiSelector method call\n *\n * @param {CssAttr} cssAttr CSS attribute object\n * @returns {string} CSS attribute parsed as UiSelector\n */\nfunction parseAttr (cssAttr) {\n  if (cssAttr.valueType && cssAttr.valueType !== 'string') {\n    throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +\n      `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);\n  }\n  const attrName = assertGetAttrName(cssAttr);\n  const methodName = toSnakeCase(attrName);\n\n  // Validate that it's a supported attribute\n  if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {\n    throw new Error(`'${attrName}' is not supported. Supported attributes are ` +\n      `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);\n  }\n\n  // Parse boolean, if it's a boolean attribute\n  if (BOOLEAN_ATTRS.includes(attrName)) {\n    return `.${methodName}(${assertGetBool(cssAttr)})`;\n  }\n\n  // Otherwise parse as string\n  let value = cssAttr.value || '';\n  if (attrName === RESOURCE_ID) {\n    value = formatIdLocator(value);\n  }\n  if (value === '') {\n    return `.${methodName}Matches(\"\")`;\n  }\n\n  switch (cssAttr.operator) {\n    case '=':\n      return `.${methodName}(\"${value}\")`;\n    case '*=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}Contains(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"${escapeRegExp(value)}\")`;\n    case '^=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}StartsWith(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"^${escapeRegExp(value)}\")`;\n    case '$=':\n      return `.${methodName}Matches(\"${escapeRegExp(value)}$\")`;\n    case '~=':\n      return `.${methodName}Matches(\"${getWordMatcherRegex(value)}\")`;\n    default:\n      // Unreachable, but adding error in case a new CSS attribute is added.\n      throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +\n        ` '=', '*=', '^=', '$=' and '~=' are supported.`);\n  }\n}\n\n/**\n * @typedef {Object} CssPseudo\n * @property {?string} valueType The type of CSS pseudo selector (https://www.npmjs.com/package/css-selector-parser for reference)\n * @property {?string} name The name of the pseudo selector\n * @property {?string} value The value of the pseudo selector\n */\n\n/**\n * Convert a CSS pseudo class to a UiSelector\n *\n * @param {CssPseudo} cssPseudo CSS Pseudo class\n * @returns {string} Pseudo selector parsed as UiSelector\n */\nfunction parsePseudo (cssPseudo) {\n  if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {\n    throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +\n      `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);\n  }\n\n  const pseudoName = assertGetAttrName(cssPseudo);\n\n  if (BOOLEAN_ATTRS.includes(pseudoName)) {\n    return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;\n  }\n\n  if (NUMERIC_ATTRS.includes(pseudoName)) {\n    return `.${pseudoName}(${cssPseudo.value})`;\n  }\n}\n\n/**\n * @typedef {Object} CssRule\n * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)\n * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)\n * @property {?string[]} classNames An array of CSS class names\n * @property {?CssAttr[]} attrs An array of CSS attributes\n * @property {?CssPseudo[]} attrs An array of CSS pseudos\n * @property {?string} id CSS identifier\n * @property {?CssRule} rule A descendant of this CSS rule\n */\n\n/**\n * Convert a CSS rule to a UiSelector\n * @param {CssRule} cssRule CSS rule definition\n */\nfunction parseCssRule (cssRule) {\n  const { nestingOperator } = cssRule;\n  if (nestingOperator && nestingOperator !== ' ') {\n    throw new Error(`'${nestingOperator}' is not a supported combinator. ` +\n      `Only child combinator (>) and descendant combinator are supported.`);\n  }\n\n  let uiAutomatorSelector = 'new UiSelector()';\n  if (cssRule.tagName && cssRule.tagName !== '*') {\n    let androidClass = [cssRule.tagName];\n    if (cssRule.classNames) {\n      for (const cssClassNames of cssRule.classNames) {\n        androidClass.push(cssClassNames);\n      }\n      uiAutomatorSelector += `.className(\"${androidClass.join('.')}\")`;\n    } else {\n      uiAutomatorSelector += `.classNameMatches(\"${cssRule.tagName}\")`;\n    }\n  } else if (cssRule.classNames) {\n    uiAutomatorSelector += `.classNameMatches(\"${cssRule.classNames.join('\\\\.')}\")`;\n  }\n  if (cssRule.id) {\n    uiAutomatorSelector += `.resourceId(\"${formatIdLocator(cssRule.id)}\")`;\n  }\n  if (cssRule.attrs) {\n    for (const attr of cssRule.attrs) {\n      uiAutomatorSelector += parseAttr(attr);\n    }\n  }\n  if (cssRule.pseudos) {\n    for (const pseudo of cssRule.pseudos) {\n      uiAutomatorSelector += parsePseudo(pseudo);\n    }\n  }\n  if (cssRule.rule) {\n    uiAutomatorSelector += `.childSelector(${parseCssRule(cssRule.rule)})`;\n  }\n  return uiAutomatorSelector;\n}\n\n/**\n * @typedef {Object} CssObject\n * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'\n */\n\n/**\n * Convert CSS object to UiAutomator2 selector\n * @param {CssObject} css CSS object\n * @returns {string} The CSS object parsed as a UiSelector\n */\nfunction parseCssObject (css) {\n  switch (css.type) {\n    case 'rule':\n      return parseCssRule(css);\n    case 'ruleSet':\n      return parseCssObject(css.rule);\n    case 'selectors':\n      return css.selectors.map((selector) => parseCssObject(selector)).join('; ');\n\n    default:\n      // This is never reachable, but if it ever is do this.\n      throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);\n  }\n}\n\n/**\n * Convert a CSS selector to a UiAutomator2 selector\n * @param {string} cssSelector CSS Selector\n * @returns {string} The CSS selector converted to a UiSelector\n */\nCssConverter.toUiAutomatorSelector = function toUiAutomatorSelector (cssSelector) {\n  let cssObj;\n  try {\n    cssObj = parser.parse(cssSelector);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Invalid CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n  try {\n    return parseCssObject(cssObj);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Unsupported CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n};\n\nexport default CssConverter;"],"file":"lib/css-converter.js","sourceRoot":"../.."}
240
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/css-converter.js"],"names":["parser","CssSelectorParser","registerSelectorPseudos","registerNestingOperators","registerAttrEqualityMods","enableSubstitutes","RESOURCE_ID","ID_LOCATOR_PATTERN","BOOLEAN_ATTRS","NUMERIC_ATTRS","STR_ATTRS","ALL_ATTRS","ATTRIBUTE_ALIASES","toSnakeCase","str","tokens","split","map","charAt","toUpperCase","slice","toLowerCase","out","join","assertGetBool","css","val","value","includes","Error","name","assertGetAttrName","attrName","officialAttr","aliasAttrs","getWordMatcherRegex","word","CssConverter","constructor","selector","pkg","formatIdLocator","locator","test","parseAttr","cssAttr","valueType","methodName","operator","parsePseudo","cssPseudo","pseudoName","parseCssRule","cssRule","nestingOperator","uiAutomatorSelector","tagName","androidClass","classNames","cssClassNames","push","id","attrs","attr","pseudos","pseudo","rule","parseCssObject","type","selectors","toUiAutomatorSelector","cssObj","parse","e","errors","InvalidSelectorError"],"mappings":"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","sourcesContent":["import { CssSelectorParser } from 'css-selector-parser';\nimport { escapeRegExp } from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst parser = new CssSelectorParser();\nparser.registerSelectorPseudos('has');\nparser.registerNestingOperators('>', '+', '~');\nparser.registerAttrEqualityMods('^', '$', '*', '~');\nparser.enableSubstitutes();\n\nconst RESOURCE_ID = 'resource-id';\nconst ID_LOCATOR_PATTERN = /^[a-zA-Z_][a-zA-Z0-9._]*:id\\/[\\S]+$/;\n\nconst BOOLEAN_ATTRS = [\n  'checkable', 'checked', 'clickable', 'enabled', 'focusable',\n  'focused', 'long-clickable', 'scrollable', 'selected',\n];\n\nconst NUMERIC_ATTRS = [\n  'index', 'instance',\n];\n\nconst STR_ATTRS = [\n  'description', RESOURCE_ID, 'text', 'class-name', 'package-name'\n];\n\nconst ALL_ATTRS = [\n  ...BOOLEAN_ATTRS,\n  ...NUMERIC_ATTRS,\n  ...STR_ATTRS,\n];\n\nconst ATTRIBUTE_ALIASES = [\n  [RESOURCE_ID, ['id']],\n  ['description', [\n    'content-description', 'content-desc',\n    'desc', 'accessibility-id',\n  ]],\n  ['index', ['nth-child']],\n];\n\n/**\n * Convert hyphen separated word to snake case\n *\n * @param {string} str\n * @returns {string} The hyphen separated word translated to snake case\n */\nfunction toSnakeCase (str) {\n  if (!str) {\n    return '';\n  }\n  const tokens = str.split('-').map((str) => str.charAt(0).toUpperCase() + str.slice(1).toLowerCase());\n  const out = tokens.join('');\n  return out.charAt(0).toLowerCase() + out.slice(1);\n}\n\n/**\n * @typedef {Object} CssNameValueObject\n * @property {?name} name The name of the CSS object\n * @property {?string} value The value of the CSS object\n */\n\n/**\n * Get the boolean from a CSS object. If empty, return true. If not true/false/empty, throw exception\n *\n * @param {CssNameValueObject} css A CSS object that has 'name' and 'value'\n * @returns {string} Either 'true' or 'false'. If value is empty, return 'true'\n */\nfunction assertGetBool (css) {\n  const val = css.value?.toLowerCase() || 'true'; // an omitted boolean attribute means 'true' (e.g.: input[checked] means checked is true)\n  if (['true', 'false'].includes(val)) {\n    return val;\n  }\n  throw new Error(`'${css.name}' must be true, false or empty. Found '${css.value}'`);\n}\n\n/**\n * Get the canonical form of a CSS attribute name\n *\n * Converts to lowercase and if an attribute name is an alias for something else, return\n * what it is an alias for\n *\n * @param {Object} css CSS object\n * @returns {string} The canonical attribute name\n */\nfunction assertGetAttrName (css) {\n  const attrName = css.name.toLowerCase();\n\n  // Check if it's supported and if it is, return it\n  if (ALL_ATTRS.includes(attrName)) {\n    return attrName.toLowerCase();\n  }\n\n  // If attrName is an alias for something else, return that\n  for (const [officialAttr, aliasAttrs] of ATTRIBUTE_ALIASES) {\n    if (aliasAttrs.includes(attrName)) {\n      return officialAttr;\n    }\n  }\n  throw new Error(`'${attrName}' is not a valid attribute. ` +\n    `Supported attributes are '${ALL_ATTRS.join(', ')}'`);\n}\n\n/**\n * Get a regex that matches a whole word. For the ~= CSS attribute selector.\n *\n * @param {string} word\n * @returns {string} A regex \"word\" matcher\n */\nfunction getWordMatcherRegex (word) {\n  return `\\\\b(\\\\w*${escapeRegExp(word)}\\\\w*)\\\\b`;\n}\n\n/**\n * @typedef {Object} CssAttr\n * @property {?string} valueType Type of attribute (must be string or empty)\n * @property {?string} value Value of the attribute\n * @property {?string} operator The operator between value and value type (=, *=, , ^=, $=)\n */\n\n/**\n * @typedef {Object} CssRule\n * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)\n * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)\n * @property {?string[]} classNames An array of CSS class names\n * @property {?CssAttr[]} attrs An array of CSS attributes\n * @property {?CssPseudo[]} attrs An array of CSS pseudos\n * @property {?string} id CSS identifier\n * @property {?CssRule} rule A descendant of this CSS rule\n */\n\n/**\n * @typedef {Object} CssObject\n * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'\n */\n\n/**\n * @typedef {Object} CssPseudo\n * @property {?string} valueType The type of CSS pseudo selector (https://www.npmjs.com/package/css-selector-parser for reference)\n * @property {?string} name The name of the pseudo selector\n * @property {?string} value The value of the pseudo selector\n */\n\nclass CssConverter {\n\n  constructor (selector, pkg) {\n    this.selector = selector;\n    this.pkg = pkg;\n  }\n\n  /**\n   * Add `<pkgName>:id/` prefix to beginning of string if it's not there already\n   *\n   * @param {string} locator The initial locator\n   * @returns {string} String with `<pkgName>:id/` prepended (if it wasn't already)\n   */\n  formatIdLocator (locator) {\n    return ID_LOCATOR_PATTERN.test(locator)\n      ? locator\n      : `${this.pkg || 'android'}:id/${locator}`;\n  }\n\n  /**\n   * Convert a CSS attribute into a UiSelector method call\n   *\n   * @param {CssAttr} cssAttr CSS attribute object\n   * @returns {string} CSS attribute parsed as UiSelector\n   */\n  parseAttr (cssAttr) {\n    if (cssAttr.valueType && cssAttr.valueType !== 'string') {\n      throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +\n        `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);\n    }\n    const attrName = assertGetAttrName(cssAttr);\n    const methodName = toSnakeCase(attrName);\n\n    // Validate that it's a supported attribute\n    if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {\n      throw new Error(`'${attrName}' is not supported. Supported attributes are ` +\n        `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);\n    }\n\n    // Parse boolean, if it's a boolean attribute\n    if (BOOLEAN_ATTRS.includes(attrName)) {\n      return `.${methodName}(${assertGetBool(cssAttr)})`;\n    }\n\n    // Otherwise parse as string\n    let value = cssAttr.value || '';\n    if (attrName === RESOURCE_ID) {\n      value = this.formatIdLocator(value);\n    }\n    if (value === '') {\n      return `.${methodName}Matches(\"\")`;\n    }\n\n    switch (cssAttr.operator) {\n      case '=':\n        return `.${methodName}(\"${value}\")`;\n      case '*=':\n        if (['description', 'text'].includes(attrName)) {\n          return `.${methodName}Contains(\"${value}\")`;\n        }\n        return `.${methodName}Matches(\"${escapeRegExp(value)}\")`;\n      case '^=':\n        if (['description', 'text'].includes(attrName)) {\n          return `.${methodName}StartsWith(\"${value}\")`;\n        }\n        return `.${methodName}Matches(\"^${escapeRegExp(value)}\")`;\n      case '$=':\n        return `.${methodName}Matches(\"${escapeRegExp(value)}$\")`;\n      case '~=':\n        return `.${methodName}Matches(\"${getWordMatcherRegex(value)}\")`;\n      default:\n        // Unreachable, but adding error in case a new CSS attribute is added.\n        throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +\n          ` '=', '*=', '^=', '$=' and '~=' are supported.`);\n    }\n  }\n\n  /**\n   * Convert a CSS pseudo class to a UiSelector\n   *\n   * @param {CssPseudo} cssPseudo CSS Pseudo class\n   * @returns {string} Pseudo selector parsed as UiSelector\n   */\n  parsePseudo (cssPseudo) {\n    if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {\n      throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +\n        `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);\n    }\n\n    const pseudoName = assertGetAttrName(cssPseudo);\n\n    if (BOOLEAN_ATTRS.includes(pseudoName)) {\n      return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;\n    }\n\n    if (NUMERIC_ATTRS.includes(pseudoName)) {\n      return `.${pseudoName}(${cssPseudo.value})`;\n    }\n  }\n\n  /**\n   * Convert a CSS rule to a UiSelector\n   * @param {CssRule} cssRule CSS rule definition\n   */\n  parseCssRule (cssRule) {\n    const { nestingOperator } = cssRule;\n    if (nestingOperator && nestingOperator !== ' ') {\n      throw new Error(`'${nestingOperator}' is not a supported combinator. ` +\n        `Only child combinator (>) and descendant combinator are supported.`);\n    }\n\n    let uiAutomatorSelector = 'new UiSelector()';\n    if (cssRule.tagName && cssRule.tagName !== '*') {\n      let androidClass = [cssRule.tagName];\n      if (cssRule.classNames) {\n        for (const cssClassNames of cssRule.classNames) {\n          androidClass.push(cssClassNames);\n        }\n        uiAutomatorSelector += `.className(\"${androidClass.join('.')}\")`;\n      } else {\n        uiAutomatorSelector += `.classNameMatches(\"${cssRule.tagName}\")`;\n      }\n    } else if (cssRule.classNames) {\n      uiAutomatorSelector += `.classNameMatches(\"${cssRule.classNames.join('\\\\.')}\")`;\n    }\n    if (cssRule.id) {\n      uiAutomatorSelector += `.resourceId(\"${this.formatIdLocator(cssRule.id)}\")`;\n    }\n    if (cssRule.attrs) {\n      for (const attr of cssRule.attrs) {\n        uiAutomatorSelector += this.parseAttr(attr);\n      }\n    }\n    if (cssRule.pseudos) {\n      for (const pseudo of cssRule.pseudos) {\n        uiAutomatorSelector += this.parsePseudo(pseudo);\n      }\n    }\n    if (cssRule.rule) {\n      uiAutomatorSelector += `.childSelector(${this.parseCssRule(cssRule.rule)})`;\n    }\n    return uiAutomatorSelector;\n  }\n\n  /**\n   * Convert CSS object to UiAutomator2 selector\n   * @param {CssObject} css CSS object\n   * @returns {string} The CSS object parsed as a UiSelector\n   */\n  parseCssObject (css) {\n    switch (css.type) {\n      case 'rule':\n        return this.parseCssRule(css);\n      case 'ruleSet':\n        return this.parseCssObject(css.rule);\n      case 'selectors':\n        return css.selectors.map((selector) => this.parseCssObject(selector)).join('; ');\n\n      default:\n        // This is never reachable, but if it ever is do this.\n        throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);\n    }\n  }\n\n  /**\n   * Convert a CSS selector to a UiAutomator2 selector\n   *\n   * @returns {string} The CSS selector converted to a UiSelector\n   */\n  toUiAutomatorSelector () {\n    let cssObj;\n    try {\n      cssObj = parser.parse(this.selector);\n    } catch (e) {\n      throw new errors.InvalidSelectorError(`Invalid CSS selector '${this.selector}'. Reason: '${e}'`);\n    }\n    try {\n      return this.parseCssObject(cssObj);\n    } catch (e) {\n      throw new errors.InvalidSelectorError(`Unsupported CSS selector '${this.selector}'. Reason: '${e}'`);\n    }\n  }\n}\n\nexport default CssConverter;"],"file":"lib/css-converter.js","sourceRoot":"../.."}
package/lib/.DS_Store ADDED
Binary file
@@ -1,6 +1,6 @@
1
1
  import CssConverter from '../css-converter';
2
2
 
3
- let helpers = {}, extensions = {};
3
+ const helpers = {};
4
4
 
5
5
  // we override the xpath search for this first-visible-child selector, which
6
6
  // looks like /*[@firstVisible="true"]
@@ -24,15 +24,11 @@ helpers.doFindElementOrEls = async function (params) {
24
24
  }
25
25
  if (params.strategy === 'css selector') {
26
26
  params.strategy = '-android uiautomator';
27
- params.selector = CssConverter.toUiAutomatorSelector(params.selector);
28
- }
29
- if (params.multiple) {
30
- return await this.uiautomator2.jwproxy.command(`/elements`, 'POST', params);
31
- } else {
32
- return await this.uiautomator2.jwproxy.command(`/element`, 'POST', params);
27
+ params.selector = new CssConverter(params.selector, this.opts.appPackage)
28
+ .toUiAutomatorSelector();
33
29
  }
30
+ return await this.uiautomator2.jwproxy.command(`/element${params.multiple ? 's' : ''}`, 'POST', params);
34
31
  };
35
32
 
36
- Object.assign(extensions, helpers);
37
33
  export { helpers };
38
- export default extensions;
34
+ export default helpers;
@@ -170,6 +170,8 @@ extensions.executeMobile = async function (mobileCommand, opts = {}) {
170
170
  installMultipleApks: 'mobileInstallMultipleApks',
171
171
 
172
172
  unlock: 'mobileUnlock',
173
+
174
+ refreshGpsCache: 'mobileRefreshGpsCache',
173
175
  };
174
176
 
175
177
  if (!_.has(mobileCommandsMapping, mobileCommand)) {
@@ -2,8 +2,6 @@ import { CssSelectorParser } from 'css-selector-parser';
2
2
  import { escapeRegExp } from 'lodash';
3
3
  import { errors } from 'appium-base-driver';
4
4
 
5
- const CssConverter = {};
6
-
7
5
  const parser = new CssSelectorParser();
8
6
  parser.registerSelectorPseudos('has');
9
7
  parser.registerNestingOperators('>', '+', '~');
@@ -113,16 +111,6 @@ function getWordMatcherRegex (word) {
113
111
  return `\\b(\\w*${escapeRegExp(word)}\\w*)\\b`;
114
112
  }
115
113
 
116
- /**
117
- * Add android:id/ to beginning of string if it's not there already
118
- *
119
- * @param {string} locator The initial locator
120
- * @returns {string} String with `android:id/` prepended (if it wasn't already)
121
- */
122
- function formatIdLocator (locator) {
123
- return ID_LOCATOR_PATTERN.test(locator) ? locator : `android:id/${locator}`;
124
- }
125
-
126
114
  /**
127
115
  * @typedef {Object} CssAttr
128
116
  * @property {?string} valueType Type of attribute (must be string or empty)
@@ -131,62 +119,20 @@ function formatIdLocator (locator) {
131
119
  */
132
120
 
133
121
  /**
134
- * Convert a CSS attribute into a UiSelector method call
135
- *
136
- * @param {CssAttr} cssAttr CSS attribute object
137
- * @returns {string} CSS attribute parsed as UiSelector
122
+ * @typedef {Object} CssRule
123
+ * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)
124
+ * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)
125
+ * @property {?string[]} classNames An array of CSS class names
126
+ * @property {?CssAttr[]} attrs An array of CSS attributes
127
+ * @property {?CssPseudo[]} attrs An array of CSS pseudos
128
+ * @property {?string} id CSS identifier
129
+ * @property {?CssRule} rule A descendant of this CSS rule
138
130
  */
139
- function parseAttr (cssAttr) {
140
- if (cssAttr.valueType && cssAttr.valueType !== 'string') {
141
- throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +
142
- `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);
143
- }
144
- const attrName = assertGetAttrName(cssAttr);
145
- const methodName = toSnakeCase(attrName);
146
-
147
- // Validate that it's a supported attribute
148
- if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {
149
- throw new Error(`'${attrName}' is not supported. Supported attributes are ` +
150
- `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);
151
- }
152
-
153
- // Parse boolean, if it's a boolean attribute
154
- if (BOOLEAN_ATTRS.includes(attrName)) {
155
- return `.${methodName}(${assertGetBool(cssAttr)})`;
156
- }
157
-
158
- // Otherwise parse as string
159
- let value = cssAttr.value || '';
160
- if (attrName === RESOURCE_ID) {
161
- value = formatIdLocator(value);
162
- }
163
- if (value === '') {
164
- return `.${methodName}Matches("")`;
165
- }
166
131
 
167
- switch (cssAttr.operator) {
168
- case '=':
169
- return `.${methodName}("${value}")`;
170
- case '*=':
171
- if (['description', 'text'].includes(attrName)) {
172
- return `.${methodName}Contains("${value}")`;
173
- }
174
- return `.${methodName}Matches("${escapeRegExp(value)}")`;
175
- case '^=':
176
- if (['description', 'text'].includes(attrName)) {
177
- return `.${methodName}StartsWith("${value}")`;
178
- }
179
- return `.${methodName}Matches("^${escapeRegExp(value)}")`;
180
- case '$=':
181
- return `.${methodName}Matches("${escapeRegExp(value)}$")`;
182
- case '~=':
183
- return `.${methodName}Matches("${getWordMatcherRegex(value)}")`;
184
- default:
185
- // Unreachable, but adding error in case a new CSS attribute is added.
186
- throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +
187
- ` '=', '*=', '^=', '$=' and '~=' are supported.`);
188
- }
189
- }
132
+ /**
133
+ * @typedef {Object} CssObject
134
+ * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'
135
+ */
190
136
 
191
137
  /**
192
138
  * @typedef {Object} CssPseudo
@@ -195,126 +141,188 @@ function parseAttr (cssAttr) {
195
141
  * @property {?string} value The value of the pseudo selector
196
142
  */
197
143
 
198
- /**
199
- * Convert a CSS pseudo class to a UiSelector
200
- *
201
- * @param {CssPseudo} cssPseudo CSS Pseudo class
202
- * @returns {string} Pseudo selector parsed as UiSelector
203
- */
204
- function parsePseudo (cssPseudo) {
205
- if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {
206
- throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +
207
- `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);
208
- }
209
-
210
- const pseudoName = assertGetAttrName(cssPseudo);
144
+ class CssConverter {
211
145
 
212
- if (BOOLEAN_ATTRS.includes(pseudoName)) {
213
- return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;
146
+ constructor (selector, pkg) {
147
+ this.selector = selector;
148
+ this.pkg = pkg;
214
149
  }
215
150
 
216
- if (NUMERIC_ATTRS.includes(pseudoName)) {
217
- return `.${pseudoName}(${cssPseudo.value})`;
151
+ /**
152
+ * Add `<pkgName>:id/` prefix to beginning of string if it's not there already
153
+ *
154
+ * @param {string} locator The initial locator
155
+ * @returns {string} String with `<pkgName>:id/` prepended (if it wasn't already)
156
+ */
157
+ formatIdLocator (locator) {
158
+ return ID_LOCATOR_PATTERN.test(locator)
159
+ ? locator
160
+ : `${this.pkg || 'android'}:id/${locator}`;
218
161
  }
219
- }
220
162
 
221
- /**
222
- * @typedef {Object} CssRule
223
- * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)
224
- * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)
225
- * @property {?string[]} classNames An array of CSS class names
226
- * @property {?CssAttr[]} attrs An array of CSS attributes
227
- * @property {?CssPseudo[]} attrs An array of CSS pseudos
228
- * @property {?string} id CSS identifier
229
- * @property {?CssRule} rule A descendant of this CSS rule
230
- */
163
+ /**
164
+ * Convert a CSS attribute into a UiSelector method call
165
+ *
166
+ * @param {CssAttr} cssAttr CSS attribute object
167
+ * @returns {string} CSS attribute parsed as UiSelector
168
+ */
169
+ parseAttr (cssAttr) {
170
+ if (cssAttr.valueType && cssAttr.valueType !== 'string') {
171
+ throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +
172
+ `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);
173
+ }
174
+ const attrName = assertGetAttrName(cssAttr);
175
+ const methodName = toSnakeCase(attrName);
231
176
 
232
- /**
233
- * Convert a CSS rule to a UiSelector
234
- * @param {CssRule} cssRule CSS rule definition
235
- */
236
- function parseCssRule (cssRule) {
237
- const { nestingOperator } = cssRule;
238
- if (nestingOperator && nestingOperator !== ' ') {
239
- throw new Error(`'${nestingOperator}' is not a supported combinator. ` +
240
- `Only child combinator (>) and descendant combinator are supported.`);
241
- }
177
+ // Validate that it's a supported attribute
178
+ if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {
179
+ throw new Error(`'${attrName}' is not supported. Supported attributes are ` +
180
+ `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);
181
+ }
242
182
 
243
- let uiAutomatorSelector = 'new UiSelector()';
244
- if (cssRule.tagName && cssRule.tagName !== '*') {
245
- let androidClass = [cssRule.tagName];
246
- if (cssRule.classNames) {
247
- for (const cssClassNames of cssRule.classNames) {
248
- androidClass.push(cssClassNames);
249
- }
250
- uiAutomatorSelector += `.className("${androidClass.join('.')}")`;
251
- } else {
252
- uiAutomatorSelector += `.classNameMatches("${cssRule.tagName}")`;
183
+ // Parse boolean, if it's a boolean attribute
184
+ if (BOOLEAN_ATTRS.includes(attrName)) {
185
+ return `.${methodName}(${assertGetBool(cssAttr)})`;
253
186
  }
254
- } else if (cssRule.classNames) {
255
- uiAutomatorSelector += `.classNameMatches("${cssRule.classNames.join('\\.')}")`;
256
- }
257
- if (cssRule.id) {
258
- uiAutomatorSelector += `.resourceId("${formatIdLocator(cssRule.id)}")`;
259
- }
260
- if (cssRule.attrs) {
261
- for (const attr of cssRule.attrs) {
262
- uiAutomatorSelector += parseAttr(attr);
187
+
188
+ // Otherwise parse as string
189
+ let value = cssAttr.value || '';
190
+ if (attrName === RESOURCE_ID) {
191
+ value = this.formatIdLocator(value);
263
192
  }
264
- }
265
- if (cssRule.pseudos) {
266
- for (const pseudo of cssRule.pseudos) {
267
- uiAutomatorSelector += parsePseudo(pseudo);
193
+ if (value === '') {
194
+ return `.${methodName}Matches("")`;
195
+ }
196
+
197
+ switch (cssAttr.operator) {
198
+ case '=':
199
+ return `.${methodName}("${value}")`;
200
+ case '*=':
201
+ if (['description', 'text'].includes(attrName)) {
202
+ return `.${methodName}Contains("${value}")`;
203
+ }
204
+ return `.${methodName}Matches("${escapeRegExp(value)}")`;
205
+ case '^=':
206
+ if (['description', 'text'].includes(attrName)) {
207
+ return `.${methodName}StartsWith("${value}")`;
208
+ }
209
+ return `.${methodName}Matches("^${escapeRegExp(value)}")`;
210
+ case '$=':
211
+ return `.${methodName}Matches("${escapeRegExp(value)}$")`;
212
+ case '~=':
213
+ return `.${methodName}Matches("${getWordMatcherRegex(value)}")`;
214
+ default:
215
+ // Unreachable, but adding error in case a new CSS attribute is added.
216
+ throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +
217
+ ` '=', '*=', '^=', '$=' and '~=' are supported.`);
268
218
  }
269
219
  }
270
- if (cssRule.rule) {
271
- uiAutomatorSelector += `.childSelector(${parseCssRule(cssRule.rule)})`;
220
+
221
+ /**
222
+ * Convert a CSS pseudo class to a UiSelector
223
+ *
224
+ * @param {CssPseudo} cssPseudo CSS Pseudo class
225
+ * @returns {string} Pseudo selector parsed as UiSelector
226
+ */
227
+ parsePseudo (cssPseudo) {
228
+ if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {
229
+ throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +
230
+ `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);
231
+ }
232
+
233
+ const pseudoName = assertGetAttrName(cssPseudo);
234
+
235
+ if (BOOLEAN_ATTRS.includes(pseudoName)) {
236
+ return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;
237
+ }
238
+
239
+ if (NUMERIC_ATTRS.includes(pseudoName)) {
240
+ return `.${pseudoName}(${cssPseudo.value})`;
241
+ }
272
242
  }
273
- return uiAutomatorSelector;
274
- }
275
243
 
276
- /**
277
- * @typedef {Object} CssObject
278
- * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'
279
- */
244
+ /**
245
+ * Convert a CSS rule to a UiSelector
246
+ * @param {CssRule} cssRule CSS rule definition
247
+ */
248
+ parseCssRule (cssRule) {
249
+ const { nestingOperator } = cssRule;
250
+ if (nestingOperator && nestingOperator !== ' ') {
251
+ throw new Error(`'${nestingOperator}' is not a supported combinator. ` +
252
+ `Only child combinator (>) and descendant combinator are supported.`);
253
+ }
280
254
 
281
- /**
282
- * Convert CSS object to UiAutomator2 selector
283
- * @param {CssObject} css CSS object
284
- * @returns {string} The CSS object parsed as a UiSelector
285
- */
286
- function parseCssObject (css) {
287
- switch (css.type) {
288
- case 'rule':
289
- return parseCssRule(css);
290
- case 'ruleSet':
291
- return parseCssObject(css.rule);
292
- case 'selectors':
293
- return css.selectors.map((selector) => parseCssObject(selector)).join('; ');
294
-
295
- default:
296
- // This is never reachable, but if it ever is do this.
297
- throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);
255
+ let uiAutomatorSelector = 'new UiSelector()';
256
+ if (cssRule.tagName && cssRule.tagName !== '*') {
257
+ let androidClass = [cssRule.tagName];
258
+ if (cssRule.classNames) {
259
+ for (const cssClassNames of cssRule.classNames) {
260
+ androidClass.push(cssClassNames);
261
+ }
262
+ uiAutomatorSelector += `.className("${androidClass.join('.')}")`;
263
+ } else {
264
+ uiAutomatorSelector += `.classNameMatches("${cssRule.tagName}")`;
265
+ }
266
+ } else if (cssRule.classNames) {
267
+ uiAutomatorSelector += `.classNameMatches("${cssRule.classNames.join('\\.')}")`;
268
+ }
269
+ if (cssRule.id) {
270
+ uiAutomatorSelector += `.resourceId("${this.formatIdLocator(cssRule.id)}")`;
271
+ }
272
+ if (cssRule.attrs) {
273
+ for (const attr of cssRule.attrs) {
274
+ uiAutomatorSelector += this.parseAttr(attr);
275
+ }
276
+ }
277
+ if (cssRule.pseudos) {
278
+ for (const pseudo of cssRule.pseudos) {
279
+ uiAutomatorSelector += this.parsePseudo(pseudo);
280
+ }
281
+ }
282
+ if (cssRule.rule) {
283
+ uiAutomatorSelector += `.childSelector(${this.parseCssRule(cssRule.rule)})`;
284
+ }
285
+ return uiAutomatorSelector;
298
286
  }
299
- }
300
287
 
301
- /**
302
- * Convert a CSS selector to a UiAutomator2 selector
303
- * @param {string} cssSelector CSS Selector
304
- * @returns {string} The CSS selector converted to a UiSelector
305
- */
306
- CssConverter.toUiAutomatorSelector = function toUiAutomatorSelector (cssSelector) {
307
- let cssObj;
308
- try {
309
- cssObj = parser.parse(cssSelector);
310
- } catch (e) {
311
- throw new errors.InvalidSelectorError(`Invalid CSS selector '${cssSelector}'. Reason: '${e}'`);
288
+ /**
289
+ * Convert CSS object to UiAutomator2 selector
290
+ * @param {CssObject} css CSS object
291
+ * @returns {string} The CSS object parsed as a UiSelector
292
+ */
293
+ parseCssObject (css) {
294
+ switch (css.type) {
295
+ case 'rule':
296
+ return this.parseCssRule(css);
297
+ case 'ruleSet':
298
+ return this.parseCssObject(css.rule);
299
+ case 'selectors':
300
+ return css.selectors.map((selector) => this.parseCssObject(selector)).join('; ');
301
+
302
+ default:
303
+ // This is never reachable, but if it ever is do this.
304
+ throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);
305
+ }
312
306
  }
313
- try {
314
- return parseCssObject(cssObj);
315
- } catch (e) {
316
- throw new errors.InvalidSelectorError(`Unsupported CSS selector '${cssSelector}'. Reason: '${e}'`);
307
+
308
+ /**
309
+ * Convert a CSS selector to a UiAutomator2 selector
310
+ *
311
+ * @returns {string} The CSS selector converted to a UiSelector
312
+ */
313
+ toUiAutomatorSelector () {
314
+ let cssObj;
315
+ try {
316
+ cssObj = parser.parse(this.selector);
317
+ } catch (e) {
318
+ throw new errors.InvalidSelectorError(`Invalid CSS selector '${this.selector}'. Reason: '${e}'`);
319
+ }
320
+ try {
321
+ return this.parseCssObject(cssObj);
322
+ } catch (e) {
323
+ throw new errors.InvalidSelectorError(`Unsupported CSS selector '${this.selector}'. Reason: '${e}'`);
324
+ }
317
325
  }
318
- };
326
+ }
319
327
 
320
328
  export default CssConverter;
package/package.json CHANGED
@@ -7,7 +7,7 @@
7
7
  "automated testing",
8
8
  "android"
9
9
  ],
10
- "version": "1.73.0",
10
+ "version": "1.74.0",
11
11
  "author": "appium",
12
12
  "license": "Apache-2.0",
13
13
  "repository": {
@@ -41,19 +41,19 @@
41
41
  ],
42
42
  "dependencies": {
43
43
  "@babel/runtime": "^7.0.0",
44
- "appium-adb": "^8.10.0",
45
- "appium-android-driver": "^4.51.0",
44
+ "appium-adb": "^8.18.0",
45
+ "appium-android-driver": "^4.54.0",
46
46
  "appium-base-driver": "^7.0.0",
47
47
  "appium-chromedriver": "^4.23.1",
48
- "appium-support": "^2.49.0",
48
+ "appium-support": "^2.54.4",
49
49
  "appium-uiautomator2-server": "^4.28.0",
50
50
  "asyncbox": "^2.3.1",
51
51
  "axios": "^0.x",
52
52
  "bluebird": "^3.5.1",
53
53
  "css-selector-parser": "^1.4.1",
54
54
  "lodash": "^4.17.4",
55
- "portscanner": "2.2.0",
56
- "source-map-support": "^0.5.5",
55
+ "portscanner": "^2.2.0",
56
+ "source-map-support": "^0.x",
57
57
  "teen_process": "^1.3.1"
58
58
  },
59
59
  "scripts": {