appium-uiautomator2-driver 1.66.1 → 1.66.2

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.
@@ -205,8 +205,8 @@ commands.mobileScroll = async function (opts = {}) {
205
205
  }
206
206
 
207
207
  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {
208
+ origin: toOrigin(elementId || element),
208
209
  params: {
209
- origin: toOrigin(elementId || element),
210
210
  strategy,
211
211
  selector,
212
212
  maxSwipes
@@ -218,4 +218,4 @@ var _default = commands;
218
218
  exports.default = _default;require('source-map-support').install();
219
219
 
220
220
 
221
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/gestures.js"],"names":["commands","toOrigin","element","util","wrapElement","unwrapElement","undefined","toPoint","x","y","_","isFinite","toRect","left","top","width","height","some","v","mobileLongClickGesture","opts","elementId","duration","uiautomator2","jwproxy","command","origin","offset","mobileDoubleClickGesture","mobileDragGesture","startX","startY","endX","endY","speed","start","end","mobileFlingGesture","direction","area","mobilePinchCloseGesture","percent","mobilePinchOpenGesture","mobileSwipeGesture","mobileScrollGesture","mobileScrollBackTo","elementToId","errors","InvalidArgumentError","mobileScroll","strategy","selector","maxSwipes","params"],"mappings":"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","sourcesContent":["import { util } from 'appium-support';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst commands = {};\n\n\nfunction toOrigin (element) {\n  return element ? util.wrapElement(util.unwrapElement(element)) : undefined;\n}\n\nfunction toPoint (x, y) {\n  return _.isFinite(x) && _.isFinite(y) ? {x, y} : undefined;\n}\n\nfunction toRect (left, top, width, height) {\n  if ([left, top, width, height].some((v) => !_.isFinite(v))) {\n    return undefined;\n  }\n  return {left, top, width, height};\n}\n\n\n/**\n * @typedef {Object} LongClickOptions\n * @property {?string} elementId - The id of the element to be clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to click on\n * @property {?number} y - The y coordinate to click on\n * @property {?number} duration [500] - Click duration in milliseconds.\n * The value must not be negative\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?LongClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileLongClickGesture = async function mobileLongClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n    duration,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/long_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n    duration,\n  });\n};\n\n/**\n * @typedef {Object} DoubleClickOptions\n * @property {?string} elementId - The id of the element to be double clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to double click on\n * @property {?number} y - The y coordinate to double click on\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?DoubleClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDoubleClickGesture = async function mobileDoubleClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/double_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n  });\n};\n\n/**\n * @typedef {Object} DragOptions\n * @property {?string} elementId - The id of the element to be dragged.\n * If the element id is missing then the start coordinates must be provided.\n * If both the element id and the start coordinates are provided then these\n * coordinates are considered as offsets from the top left element corner.\n * @property {?number} startX - The x coordinate where the dragging starts\n * @property {?number} startY - The y coordinate where the dragging starts\n * @property {!number} endX - The x coordinate where the dragging ends\n * @property {!number} endY - The y coordinate where the dragging ends\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Drags this object to the specified location.\n *\n * @param {?DragOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDragGesture = async function mobileDragGesture (opts = {}) {\n  const {\n    elementId,\n    startX, startY,\n    endX, endY,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/drag', 'POST', {\n    origin: toOrigin(elementId),\n    start: toPoint(startX, startY),\n    end: toPoint(endX, endY),\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} FlingOptions\n * @property {?string} elementId - The id of the element to be flinged.\n * If the element id is missing then fling bounding area must be provided.\n * If both the element id and the fling bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the fling bounding area\n * @property {?number} top - The top coordinate of the fling bounding area\n * @property {?number} width - The width of the fling bounding area\n * @property {?number} height - The height of the fling bounding area\n * @property {!string} direction - Direction of the fling.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {?number} speed [7500 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must be greater than the minimum fling\n * velocity for the given view (50 by default)\n */\n\n/**\n * Drags to the specified location.\n *\n * @param {?FlingOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileFlingGesture = async function mobileFlingGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/fling', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} PinchOptions\n * @property {?string} elementId - The id of the element to be pinched.\n * If the element id is missing then pinch bounding area must be provided.\n * If both the element id and the pinch bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the pinch bounding area\n * @property {?number} top - The top coordinate of the pinch bounding area\n * @property {?number} width - The width of the pinch bounding area\n * @property {?number} height - The height of the pinch bounding area\n * @property {!number} percent - The size of the pinch as a percentage of the pinch area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a pinch close gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchCloseGesture = async function mobilePinchCloseGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_close', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * Performs a pinch open gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchOpenGesture = async function mobilePinchOpenGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_open', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} SwipeOptions\n * @property {?string} elementId - The id of the element to be swiped.\n * If the element id is missing then swipe bounding area must be provided.\n * If both the element id and the swipe bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the swipe bounding area\n * @property {?number} top - The top coordinate of the swipe bounding area\n * @property {?number} width - The width of the swipe bounding area\n * @property {?number} height - The height of the swipe bounding area\n * @property {!string} direction - Direction of the swipe.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the swipe as a percentage of the swipe area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a swipe gesture.\n *\n * @param {?SwipeOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileSwipeGesture = async function mobileSwipeGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/swipe', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollOptions\n * @property {?string} elementId - The id of the element to be scrolled.\n * If the element id is missing then scroll bounding area must be provided.\n * If both the element id and the scroll bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the scroll bounding area\n * @property {?number} top - The top coordinate of the scroll bounding area\n * @property {?number} width - The width of the scroll bounding area\n * @property {?number} height - The height of the scroll bounding area\n * @property {!string} direction - Direction of the scroll.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the scroll as a percentage of the scrolling area size.\n * Valid values must be float numbers greater than zero, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this gesture\n * in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a scroll gesture.\n *\n * @param {?ScrollOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileScrollGesture = async function mobileScrollGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/scroll', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollElementToElementOpts\n * @property {string} elementId The identifier of the scrollable element,\n * which is going to be scrolled. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy.\n * @property {string} elementToId The identifier of the item, which belongs\n * to the scrollable element above, and which should become visible after\n * the scrolling operation is finished. It is required this element\n * was located by `-android uiautomator` strategy.\n */\n\n/**\n * Scrolls the given scrollable element `elementId` until `elementToId`\n * becomes visible. This function returns immediately if the `elementToId`\n * is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollElementToElementOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScrollBackTo = async function (opts = {}) {\n  const {elementId, elementToId} = opts;\n  if (!elementId || !elementToId) {\n    throw new errors.InvalidArgumentError(`Both elementId and elementToId arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command(\n    `/appium/element/${util.unwrapElement(elementId)}/scroll_to/${util.unwrapElement(elementToId)}`, 'POST', {});\n};\n\n/**\n * @typedef {Object} ScrollOpts\n * @property {?string} elementId The identifier of an element. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy. If this property is not provided then the first currently available scrollable view\n * is selected for the interaction.\n * @property {!string} strategy The following strategies are supported:\n * - `accessibility id` (UiSelector().description)\n * - `class name` (UiSelector().className)\n * - `-android uiautomator` (UiSelector)\n * @property {!string} selector The corresponding lookup value for the given\n * strategy.\n * @property {?number} maxSwipes The maximum number of swipes to perform\n * on the target scrollable view in order to reach the destination element.\n * In case this value is unset then it would be retrieved from the scrollable\n * element itself (vua `getMaxSearchSwipes()` property).\n */\n\n/**\n * Scrolls the given scrollable element until the element identified\n * by `strategy` and `selector` becomes visible. This function returns immediately if the\n * destination element is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScroll = async function (opts = {}) {\n  const {\n    element, elementId, // `element` is deprecated, use `elementId` instead\n    strategy, selector, maxSwipes\n  } = opts;\n  if (!strategy || !selector) {\n    throw new errors.InvalidArgumentError(`Both strategy and selector arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {\n    params: {\n      origin: toOrigin(elementId || element),\n      strategy, selector, maxSwipes\n    },\n  });\n};\n\nexport default commands;\n"],"file":"lib/commands/gestures.js","sourceRoot":"../../.."}
221
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/gestures.js"],"names":["commands","toOrigin","element","util","wrapElement","unwrapElement","undefined","toPoint","x","y","_","isFinite","toRect","left","top","width","height","some","v","mobileLongClickGesture","opts","elementId","duration","uiautomator2","jwproxy","command","origin","offset","mobileDoubleClickGesture","mobileDragGesture","startX","startY","endX","endY","speed","start","end","mobileFlingGesture","direction","area","mobilePinchCloseGesture","percent","mobilePinchOpenGesture","mobileSwipeGesture","mobileScrollGesture","mobileScrollBackTo","elementToId","errors","InvalidArgumentError","mobileScroll","strategy","selector","maxSwipes","params"],"mappings":"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","sourcesContent":["import { util } from 'appium-support';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst commands = {};\n\n\nfunction toOrigin (element) {\n  return element ? util.wrapElement(util.unwrapElement(element)) : undefined;\n}\n\nfunction toPoint (x, y) {\n  return _.isFinite(x) && _.isFinite(y) ? {x, y} : undefined;\n}\n\nfunction toRect (left, top, width, height) {\n  if ([left, top, width, height].some((v) => !_.isFinite(v))) {\n    return undefined;\n  }\n  return {left, top, width, height};\n}\n\n\n/**\n * @typedef {Object} LongClickOptions\n * @property {?string} elementId - The id of the element to be clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to click on\n * @property {?number} y - The y coordinate to click on\n * @property {?number} duration [500] - Click duration in milliseconds.\n * The value must not be negative\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?LongClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileLongClickGesture = async function mobileLongClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n    duration,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/long_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n    duration,\n  });\n};\n\n/**\n * @typedef {Object} DoubleClickOptions\n * @property {?string} elementId - The id of the element to be double clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to double click on\n * @property {?number} y - The y coordinate to double click on\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?DoubleClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDoubleClickGesture = async function mobileDoubleClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/double_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n  });\n};\n\n/**\n * @typedef {Object} DragOptions\n * @property {?string} elementId - The id of the element to be dragged.\n * If the element id is missing then the start coordinates must be provided.\n * If both the element id and the start coordinates are provided then these\n * coordinates are considered as offsets from the top left element corner.\n * @property {?number} startX - The x coordinate where the dragging starts\n * @property {?number} startY - The y coordinate where the dragging starts\n * @property {!number} endX - The x coordinate where the dragging ends\n * @property {!number} endY - The y coordinate where the dragging ends\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Drags this object to the specified location.\n *\n * @param {?DragOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDragGesture = async function mobileDragGesture (opts = {}) {\n  const {\n    elementId,\n    startX, startY,\n    endX, endY,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/drag', 'POST', {\n    origin: toOrigin(elementId),\n    start: toPoint(startX, startY),\n    end: toPoint(endX, endY),\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} FlingOptions\n * @property {?string} elementId - The id of the element to be flinged.\n * If the element id is missing then fling bounding area must be provided.\n * If both the element id and the fling bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the fling bounding area\n * @property {?number} top - The top coordinate of the fling bounding area\n * @property {?number} width - The width of the fling bounding area\n * @property {?number} height - The height of the fling bounding area\n * @property {!string} direction - Direction of the fling.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {?number} speed [7500 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must be greater than the minimum fling\n * velocity for the given view (50 by default)\n */\n\n/**\n * Drags to the specified location.\n *\n * @param {?FlingOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileFlingGesture = async function mobileFlingGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/fling', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} PinchOptions\n * @property {?string} elementId - The id of the element to be pinched.\n * If the element id is missing then pinch bounding area must be provided.\n * If both the element id and the pinch bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the pinch bounding area\n * @property {?number} top - The top coordinate of the pinch bounding area\n * @property {?number} width - The width of the pinch bounding area\n * @property {?number} height - The height of the pinch bounding area\n * @property {!number} percent - The size of the pinch as a percentage of the pinch area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a pinch close gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchCloseGesture = async function mobilePinchCloseGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_close', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * Performs a pinch open gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchOpenGesture = async function mobilePinchOpenGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_open', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} SwipeOptions\n * @property {?string} elementId - The id of the element to be swiped.\n * If the element id is missing then swipe bounding area must be provided.\n * If both the element id and the swipe bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the swipe bounding area\n * @property {?number} top - The top coordinate of the swipe bounding area\n * @property {?number} width - The width of the swipe bounding area\n * @property {?number} height - The height of the swipe bounding area\n * @property {!string} direction - Direction of the swipe.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the swipe as a percentage of the swipe area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a swipe gesture.\n *\n * @param {?SwipeOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileSwipeGesture = async function mobileSwipeGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/swipe', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollOptions\n * @property {?string} elementId - The id of the element to be scrolled.\n * If the element id is missing then scroll bounding area must be provided.\n * If both the element id and the scroll bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the scroll bounding area\n * @property {?number} top - The top coordinate of the scroll bounding area\n * @property {?number} width - The width of the scroll bounding area\n * @property {?number} height - The height of the scroll bounding area\n * @property {!string} direction - Direction of the scroll.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the scroll as a percentage of the scrolling area size.\n * Valid values must be float numbers greater than zero, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this gesture\n * in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a scroll gesture.\n *\n * @param {?ScrollOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileScrollGesture = async function mobileScrollGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/scroll', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollElementToElementOpts\n * @property {string} elementId The identifier of the scrollable element,\n * which is going to be scrolled. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy.\n * @property {string} elementToId The identifier of the item, which belongs\n * to the scrollable element above, and which should become visible after\n * the scrolling operation is finished. It is required this element\n * was located by `-android uiautomator` strategy.\n */\n\n/**\n * Scrolls the given scrollable element `elementId` until `elementToId`\n * becomes visible. This function returns immediately if the `elementToId`\n * is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollElementToElementOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScrollBackTo = async function (opts = {}) {\n  const {elementId, elementToId} = opts;\n  if (!elementId || !elementToId) {\n    throw new errors.InvalidArgumentError(`Both elementId and elementToId arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command(\n    `/appium/element/${util.unwrapElement(elementId)}/scroll_to/${util.unwrapElement(elementToId)}`, 'POST', {});\n};\n\n/**\n * @typedef {Object} ScrollOpts\n * @property {?string} elementId The identifier of an element. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy. If this property is not provided then the first currently available scrollable view\n * is selected for the interaction.\n * @property {!string} strategy The following strategies are supported:\n * - `accessibility id` (UiSelector().description)\n * - `class name` (UiSelector().className)\n * - `-android uiautomator` (UiSelector)\n * @property {!string} selector The corresponding lookup value for the given\n * strategy.\n * @property {?number} maxSwipes The maximum number of swipes to perform\n * on the target scrollable view in order to reach the destination element.\n * In case this value is unset then it would be retrieved from the scrollable\n * element itself (vua `getMaxSearchSwipes()` property).\n */\n\n/**\n * Scrolls the given scrollable element until the element identified\n * by `strategy` and `selector` becomes visible. This function returns immediately if the\n * destination element is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScroll = async function (opts = {}) {\n  const {\n    element, elementId, // `element` is deprecated, use `elementId` instead\n    strategy, selector, maxSwipes\n  } = opts;\n  if (!strategy || !selector) {\n    throw new errors.InvalidArgumentError(`Both strategy and selector arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {\n    origin: toOrigin(elementId || element),\n    params: {strategy, selector, maxSwipes},\n  });\n};\n\nexport default commands;\n"],"file":"lib/commands/gestures.js","sourceRoot":"../../.."}
@@ -349,12 +349,14 @@ class UiAutomator2Server {
349
349
  url: `http://${this.host}:${this.systemPort}/wd/hub/sessions`,
350
350
  timeout: 500
351
351
  })).data;
352
- const activeSessionIds = value.map(sess => sess.id);
352
+ const activeSessionIds = value.map(({
353
+ id
354
+ }) => id).filter(Boolean);
353
355
 
354
356
  if (activeSessionIds.length) {
355
357
  _logger.default.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);
356
358
 
357
- _logger.default.debug('Cleaning up the obsolete sessions');
359
+ _logger.default.debug(`Cleaning up ${_appiumSupport.util.pluralize('obsolete session', activeSessionIds.length, true)}`);
358
360
 
359
361
  await _bluebird.default.all(activeSessionIds.map(id => _axios.default.delete(`http://${this.host}:${this.systemPort}/wd/hub/session/${id}`)));
360
362
  await _bluebird.default.delay(1000);
@@ -385,4 +387,4 @@ var _default = UiAutomator2Server;
385
387
  exports.default = _default;require('source-map-support').install();
386
388
 
387
389
 
388
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/uiautomator2.js"],"names":["REQD_PARAMS","SERVER_LAUNCH_TIMEOUT","SERVER_INSTALL_RETRIES","SERVICES_LAUNCH_TIMEOUT","SERVER_PACKAGE_ID","SERVER_TEST_PACKAGE_ID","INSTRUMENTATION_TARGET","instrumentationLogger","logger","getLogger","UIA2Proxy","JWProxy","proxyCommand","url","method","body","didInstrumentationExit","errors","InvalidContextError","UiAutomator2Server","constructor","opts","req","util","hasValue","Error","disableSuppressAccessibilityService","proxyOpts","server","host","port","systemPort","keepAlive","readTimeout","timeout","jwproxy","proxyReqRes","bind","installServerApk","installTimeout","tmpRoot","tempDir","openDir","packageInfosMapper","appPath","appId","helpers","isWriteable","log","info","dstPath","path","resolve","basename","fs","copyFile","packagesInfo","B","all","map","apkPath","testApkPath","shouldUninstallServerPackages","shouldInstallServerPackages","isAppInstalled","adb","checkApkCert","signApp","appState","getApplicationInstallState","debug","APP_INSTALL_STATE","OLDER_VERSION_INSTALLED","NEWER_VERSION_INSTALLED","includes","NOT_INSTALLED","uninstallApk","err","warn","message","install","noIncremental","replace","timeoutCapName","rimraf","verifyServicesAvailability","isPmServiceAvailable","pmOutput","pmError","shell","e","waitMs","intervalMs","error","line","split","startSession","caps","cleanupAutomationLeftovers","skipServerInstallation","serverVersion","uiautomator2ServerLaunchTimeout","timer","timing","Timer","start","retries","maxRetries","delayBetweenRetries","startInstrumentationProcess","command","errorAndThrow","delay","getDuration","asMilliSeconds","toFixed","capabilities","firstMatch","alwaysMatch","cmd","disableWindowAnimation","push","_","isBoolean","instrumentationProcess","createSubProcess","on","stdout","stderr","output","trim","code","deleteSession","strictCleanup","value","data","activeSessionIds","sess","id","length","JSON","stringify","axios","delete","forceStop","ignore","killProcessesByName"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { JWProxy, errors } from 'appium-base-driver';\nimport { waitForCondition } from 'asyncbox';\nimport log from './logger';\nimport { SERVER_APK_PATH as apkPath, TEST_APK_PATH as testApkPath, version as serverVersion } from 'appium-uiautomator2-server';\nimport { util, logger, tempDir, fs, timing } from 'appium-support';\nimport B from 'bluebird';\nimport helpers from './helpers';\nimport axios from 'axios';\nimport path from 'path';\n\nconst REQD_PARAMS = ['adb', 'tmpDir', 'host', 'systemPort', 'devicePort', 'disableWindowAnimation'];\nconst SERVER_LAUNCH_TIMEOUT = 30000;\nconst SERVER_INSTALL_RETRIES = 20;\nconst SERVICES_LAUNCH_TIMEOUT = 30000;\nconst SERVER_PACKAGE_ID = 'io.appium.uiautomator2.server';\nconst SERVER_TEST_PACKAGE_ID = `${SERVER_PACKAGE_ID}.test`;\nconst INSTRUMENTATION_TARGET = `${SERVER_TEST_PACKAGE_ID}/androidx.test.runner.AndroidJUnitRunner`;\nconst instrumentationLogger = logger.getLogger('Instrumentation');\n\nclass UIA2Proxy extends JWProxy {\n  async proxyCommand (url, method, body = null) {\n    if (this.didInstrumentationExit) {\n      throw new errors.InvalidContextError(\n        `'${method} ${url}' cannot be proxied to UiAutomator2 server because ` +\n        'the instrumentation process is not running (probably crashed). ' +\n        'Check the server log and/or the logcat output for more details');\n    }\n    return await super.proxyCommand(url, method, body);\n  }\n}\n\nclass UiAutomator2Server {\n  constructor (opts = {}) {\n    for (let req of REQD_PARAMS) {\n      if (!opts || !util.hasValue(opts[req])) {\n        throw new Error(`Option '${req}' is required!`);\n      }\n      this[req] = opts[req];\n    }\n    this.disableSuppressAccessibilityService = opts.disableSuppressAccessibilityService;\n    const proxyOpts = {\n      server: this.host,\n      port: this.systemPort,\n      keepAlive: true,\n    };\n    if (opts.readTimeout && opts.readTimeout > 0) {\n      proxyOpts.timeout = opts.readTimeout;\n    }\n    this.jwproxy = new UIA2Proxy(proxyOpts);\n    this.proxyReqRes = this.jwproxy.proxyReqRes.bind(this.jwproxy);\n    this.jwproxy.didInstrumentationExit = false;\n  }\n\n  /**\n   * Installs the apks on to the device or emulator.\n   *\n   * @param {number} installTimeout - Installation timeout\n   */\n  async installServerApk (installTimeout = SERVER_INSTALL_RETRIES * 1000) {\n    const tmpRoot = await tempDir.openDir();\n    const packageInfosMapper = async ({appPath, appId}) => {\n      if (await helpers.isWriteable(appPath)) {\n        return { appPath, appId };\n      }\n\n      log.info(`Server package at '${appPath}' is not writeable. ` +\n        `Will copy it into the temporary location at '${tmpRoot}' as a workaround. ` +\n        `Consider making this file writeable manually in order to improve the performance of session startup.`);\n      const dstPath = path.resolve(tmpRoot, path.basename(appPath));\n      await fs.copyFile(appPath, dstPath);\n      return {\n        appPath: dstPath,\n        appId,\n      };\n    };\n\n    try {\n      const packagesInfo = await B.all(B.map([\n        {\n          appPath: apkPath,\n          appId: SERVER_PACKAGE_ID,\n        }, {\n          appPath: testApkPath,\n          appId: SERVER_TEST_PACKAGE_ID,\n        },\n      ], packageInfosMapper));\n\n      let shouldUninstallServerPackages = false;\n      let shouldInstallServerPackages = false;\n      for (const {appId, appPath} of packagesInfo) {\n        if (appId === SERVER_TEST_PACKAGE_ID) {\n          const isAppInstalled = await this.adb.isAppInstalled(appId);\n\n          // There is no point in getting the state for test server,\n          // since it does not contain version info\n          if (!await this.adb.checkApkCert(appPath, appId)) {\n            await helpers.signApp(this.adb, appPath);\n            shouldUninstallServerPackages = shouldUninstallServerPackages || isAppInstalled;\n            shouldInstallServerPackages = true;\n          }\n\n          if (!isAppInstalled) {\n            shouldInstallServerPackages = true;\n          }\n          continue;\n        }\n\n        const appState = await this.adb.getApplicationInstallState(appPath, appId);\n        log.debug(`${appId} installation state: ${appState}`);\n        if (await this.adb.checkApkCert(appPath, appId)) {\n          shouldUninstallServerPackages = shouldUninstallServerPackages || [\n            this.adb.APP_INSTALL_STATE.OLDER_VERSION_INSTALLED,\n            this.adb.APP_INSTALL_STATE.NEWER_VERSION_INSTALLED,\n          ].includes(appState);\n        } else {\n          await helpers.signApp(this.adb, appPath);\n          shouldUninstallServerPackages = shouldUninstallServerPackages || ![\n            this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n          ].includes(appState);\n        }\n        shouldInstallServerPackages = shouldInstallServerPackages || shouldUninstallServerPackages || [\n          this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n        ].includes(appState);\n      }\n      log.info(`Server packages are ${shouldInstallServerPackages ? '' : 'not '}going to be (re)installed`);\n      if (shouldInstallServerPackages && shouldUninstallServerPackages) {\n        log.info('Full packages reinstall is going to be performed');\n      }\n      for (const {appId, appPath} of packagesInfo) {\n        if (shouldUninstallServerPackages) {\n          try {\n            await this.adb.uninstallApk(appId);\n          } catch (err) {\n            log.warn(`Error uninstalling '${appId}': ${err.message}`);\n          }\n        }\n        if (shouldInstallServerPackages) {\n          await this.adb.install(appPath, {\n            noIncremental: true,\n            replace: true,\n            timeout: installTimeout,\n            timeoutCapName: 'uiautomator2ServerInstallTimeout'\n          });\n        }\n      }\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n\n    await this.verifyServicesAvailability();\n  }\n\n  async verifyServicesAvailability () {\n    log.debug(`Waiting up to ${SERVICES_LAUNCH_TIMEOUT}ms for services to be available`);\n    let isPmServiceAvailable = false;\n    let pmOutput = '';\n    let pmError = null;\n    try {\n      await waitForCondition(async () => {\n        if (!isPmServiceAvailable) {\n          pmError = null;\n          pmOutput = '';\n          try {\n            pmOutput = await this.adb.shell(['pm', 'list', 'instrumentation']);\n          } catch (e) {\n            pmError = e;\n          }\n          if (pmOutput.includes('Could not access the Package Manager')) {\n            pmError = new Error(`Problem running Package Manager: ${pmOutput}`);\n            pmOutput = ''; // remove output, so it is not printed below\n          } else if (pmOutput.includes(INSTRUMENTATION_TARGET)) {\n            pmOutput = ''; // remove output, so it is not printed below\n            log.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);\n            // eslint-disable-next-line require-atomic-updates\n            isPmServiceAvailable = true;\n          } else if (!pmError) {\n            pmError = new Error('The instrumentation target is not listed by Package Manager');\n          }\n        }\n        return isPmServiceAvailable;\n      }, {\n        waitMs: SERVICES_LAUNCH_TIMEOUT,\n        intervalMs: 1000,\n      });\n    } catch (err) {\n      log.error(`Unable to find instrumentation target '${INSTRUMENTATION_TARGET}': ${(pmError || {}).message}`);\n      if (pmOutput) {\n        log.debug('Available targets:');\n        for (const line of pmOutput.split('\\n')) {\n          log.debug(`    ${line.replace('instrumentation:', '')}`);\n        }\n      }\n    }\n  }\n\n  async startSession (caps) {\n    await this.cleanupAutomationLeftovers();\n    if (caps.skipServerInstallation) {\n      log.info(`'skipServerInstallation' is set. Attempting to use UIAutomator2 server from the device`);\n    } else {\n      log.info(`Starting UIAutomator2 server ${serverVersion}`);\n      log.info(`Using UIAutomator2 server from '${apkPath}' and test from '${testApkPath}'`);\n    }\n\n    const timeout = caps.uiautomator2ServerLaunchTimeout || SERVER_LAUNCH_TIMEOUT;\n    const timer = new timing.Timer().start();\n    let retries = 0;\n    const maxRetries = 2;\n    const delayBetweenRetries = 3000;\n    while (retries < maxRetries) {\n      log.info(`Waiting up to ${timeout}ms for UiAutomator2 to be online...`);\n      this.jwproxy.didInstrumentationExit = false;\n      await this.startInstrumentationProcess();\n      if (!this.jwproxy.didInstrumentationExit) {\n        try {\n          await waitForCondition(async () => {\n            try {\n              await this.jwproxy.command('/status', 'GET');\n              return true;\n            } catch (err) {\n              // short circuit to retry or fail fast\n              return this.jwproxy.didInstrumentationExit;\n            }\n          }, {\n            waitMs: timeout,\n            intervalMs: 1000,\n          });\n        } catch (err) {\n          log.errorAndThrow(`The instrumentation process cannot be initialized within ${timeout}ms timeout. `\n            + 'Make sure the application under test does not crash and investigate the logcat output. '\n            + `You could also try to increase the value of 'uiautomator2ServerLaunchTimeout' capability`);\n        }\n      }\n      if (!this.jwproxy.didInstrumentationExit) {\n        break;\n      }\n\n      retries++;\n      if (retries >= maxRetries) {\n        log.errorAndThrow('The instrumentation process cannot be initialized. '\n          + 'Make sure the application under test does not crash and investigate the logcat output.');\n      }\n      log.warn(`The instrumentation process has been unexpectedly terminated. `\n        + `Retrying UiAutomator2 startup (#${retries} of ${maxRetries - 1})`);\n      await this.cleanupAutomationLeftovers(true);\n      await B.delay(delayBetweenRetries);\n    }\n\n    log.debug(`The initialization of the instrumentation process took `\n      + `${timer.getDuration().asMilliSeconds.toFixed(0)}ms`);\n    await this.jwproxy.command('/session', 'POST', {\n      capabilities: {\n        firstMatch: [caps],\n        alwaysMatch: {},\n      }\n    });\n  }\n\n  async startInstrumentationProcess () {\n    const cmd = ['am', 'instrument', '-w'];\n    if (this.disableWindowAnimation) {\n      cmd.push('--no-window-animation');\n    }\n    if (_.isBoolean(this.disableSuppressAccessibilityService)) {\n      cmd.push('-e', 'DISABLE_SUPPRESS_ACCESSIBILITY_SERVICES', this.disableSuppressAccessibilityService);\n    }\n    // Disable Google analytics to prevent possible fatal exception\n    cmd.push('-e', 'disableAnalytics', true);\n    cmd.push(INSTRUMENTATION_TARGET);\n    const instrumentationProcess = this.adb.createSubProcess(['shell', ...cmd]);\n    instrumentationProcess.on('output', (stdout, stderr) => {\n      const output = _.trim(stdout || stderr);\n      if (output) {\n        instrumentationLogger.debug(output);\n      }\n    });\n    instrumentationProcess.on('exit', (code) => {\n      instrumentationLogger.debug(`The process has exited with code ${code}`);\n      this.jwproxy.didInstrumentationExit = true;\n    });\n    await instrumentationProcess.start(0);\n  }\n\n  async deleteSession () {\n    log.debug('Deleting UiAutomator2 server session');\n    // rely on jwproxy's intelligence to know what we're talking about and\n    // delete the current session\n    try {\n      await this.jwproxy.command('/', 'DELETE');\n    } catch (err) {\n      log.warn(`Did not get confirmation UiAutomator2 deleteSession worked; ` +\n          `Error was: ${err}`);\n    }\n  }\n\n  async cleanupAutomationLeftovers (strictCleanup = false) {\n    log.debug(`Performing ${strictCleanup ? 'strict' : 'shallow'} cleanup of automation leftovers`);\n\n    try {\n      const {value} = (await axios({\n        url: `http://${this.host}:${this.systemPort}/wd/hub/sessions`,\n        timeout: 500,\n      })).data;\n      const activeSessionIds = value.map((sess) => sess.id);\n      if (activeSessionIds.length) {\n        log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);\n        log.debug('Cleaning up the obsolete sessions');\n        await B.all(activeSessionIds.map((id) =>\n          axios.delete(`http://${this.host}:${this.systemPort}/wd/hub/session/${id}`)\n        ));\n        // Let all sessions to be properly terminated before continuing\n        await B.delay(1000);\n      } else {\n        log.debug('No obsolete sessions have been detected');\n      }\n    } catch (e) {\n      log.debug(`No obsolete sessions have been detected (${e.message})`);\n    }\n\n    try {\n      await this.adb.forceStop(SERVER_TEST_PACKAGE_ID);\n    } catch (ignore) {}\n    if (!strictCleanup) {\n      return;\n    }\n    // https://github.com/appium/appium/issues/10749\n    try {\n      await this.adb.killProcessesByName('uiautomator');\n    } catch (ignore) {}\n  }\n}\n\nexport { UiAutomator2Server, INSTRUMENTATION_TARGET, SERVER_PACKAGE_ID, SERVER_TEST_PACKAGE_ID };\nexport default UiAutomator2Server;\n"],"file":"lib/uiautomator2.js","sourceRoot":"../.."}
390
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/uiautomator2.js"],"names":["REQD_PARAMS","SERVER_LAUNCH_TIMEOUT","SERVER_INSTALL_RETRIES","SERVICES_LAUNCH_TIMEOUT","SERVER_PACKAGE_ID","SERVER_TEST_PACKAGE_ID","INSTRUMENTATION_TARGET","instrumentationLogger","logger","getLogger","UIA2Proxy","JWProxy","proxyCommand","url","method","body","didInstrumentationExit","errors","InvalidContextError","UiAutomator2Server","constructor","opts","req","util","hasValue","Error","disableSuppressAccessibilityService","proxyOpts","server","host","port","systemPort","keepAlive","readTimeout","timeout","jwproxy","proxyReqRes","bind","installServerApk","installTimeout","tmpRoot","tempDir","openDir","packageInfosMapper","appPath","appId","helpers","isWriteable","log","info","dstPath","path","resolve","basename","fs","copyFile","packagesInfo","B","all","map","apkPath","testApkPath","shouldUninstallServerPackages","shouldInstallServerPackages","isAppInstalled","adb","checkApkCert","signApp","appState","getApplicationInstallState","debug","APP_INSTALL_STATE","OLDER_VERSION_INSTALLED","NEWER_VERSION_INSTALLED","includes","NOT_INSTALLED","uninstallApk","err","warn","message","install","noIncremental","replace","timeoutCapName","rimraf","verifyServicesAvailability","isPmServiceAvailable","pmOutput","pmError","shell","e","waitMs","intervalMs","error","line","split","startSession","caps","cleanupAutomationLeftovers","skipServerInstallation","serverVersion","uiautomator2ServerLaunchTimeout","timer","timing","Timer","start","retries","maxRetries","delayBetweenRetries","startInstrumentationProcess","command","errorAndThrow","delay","getDuration","asMilliSeconds","toFixed","capabilities","firstMatch","alwaysMatch","cmd","disableWindowAnimation","push","_","isBoolean","instrumentationProcess","createSubProcess","on","stdout","stderr","output","trim","code","deleteSession","strictCleanup","value","data","activeSessionIds","id","filter","Boolean","length","JSON","stringify","pluralize","axios","delete","forceStop","ignore","killProcessesByName"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport { JWProxy, errors } from 'appium-base-driver';\nimport { waitForCondition } from 'asyncbox';\nimport log from './logger';\nimport {\n  SERVER_APK_PATH as apkPath,\n  TEST_APK_PATH as testApkPath,\n  version as serverVersion\n} from 'appium-uiautomator2-server';\nimport {\n  util, logger, tempDir, fs, timing\n} from 'appium-support';\nimport B from 'bluebird';\nimport helpers from './helpers';\nimport axios from 'axios';\nimport path from 'path';\n\nconst REQD_PARAMS = ['adb', 'tmpDir', 'host', 'systemPort', 'devicePort', 'disableWindowAnimation'];\nconst SERVER_LAUNCH_TIMEOUT = 30000;\nconst SERVER_INSTALL_RETRIES = 20;\nconst SERVICES_LAUNCH_TIMEOUT = 30000;\nconst SERVER_PACKAGE_ID = 'io.appium.uiautomator2.server';\nconst SERVER_TEST_PACKAGE_ID = `${SERVER_PACKAGE_ID}.test`;\nconst INSTRUMENTATION_TARGET = `${SERVER_TEST_PACKAGE_ID}/androidx.test.runner.AndroidJUnitRunner`;\nconst instrumentationLogger = logger.getLogger('Instrumentation');\n\nclass UIA2Proxy extends JWProxy {\n  async proxyCommand (url, method, body = null) {\n    if (this.didInstrumentationExit) {\n      throw new errors.InvalidContextError(\n        `'${method} ${url}' cannot be proxied to UiAutomator2 server because ` +\n        'the instrumentation process is not running (probably crashed). ' +\n        'Check the server log and/or the logcat output for more details');\n    }\n    return await super.proxyCommand(url, method, body);\n  }\n}\n\nclass UiAutomator2Server {\n  constructor (opts = {}) {\n    for (let req of REQD_PARAMS) {\n      if (!opts || !util.hasValue(opts[req])) {\n        throw new Error(`Option '${req}' is required!`);\n      }\n      this[req] = opts[req];\n    }\n    this.disableSuppressAccessibilityService = opts.disableSuppressAccessibilityService;\n    const proxyOpts = {\n      server: this.host,\n      port: this.systemPort,\n      keepAlive: true,\n    };\n    if (opts.readTimeout && opts.readTimeout > 0) {\n      proxyOpts.timeout = opts.readTimeout;\n    }\n    this.jwproxy = new UIA2Proxy(proxyOpts);\n    this.proxyReqRes = this.jwproxy.proxyReqRes.bind(this.jwproxy);\n    this.jwproxy.didInstrumentationExit = false;\n  }\n\n  /**\n   * Installs the apks on to the device or emulator.\n   *\n   * @param {number} installTimeout - Installation timeout\n   */\n  async installServerApk (installTimeout = SERVER_INSTALL_RETRIES * 1000) {\n    const tmpRoot = await tempDir.openDir();\n    const packageInfosMapper = async ({appPath, appId}) => {\n      if (await helpers.isWriteable(appPath)) {\n        return { appPath, appId };\n      }\n\n      log.info(`Server package at '${appPath}' is not writeable. ` +\n        `Will copy it into the temporary location at '${tmpRoot}' as a workaround. ` +\n        `Consider making this file writeable manually in order to improve the performance of session startup.`);\n      const dstPath = path.resolve(tmpRoot, path.basename(appPath));\n      await fs.copyFile(appPath, dstPath);\n      return {\n        appPath: dstPath,\n        appId,\n      };\n    };\n\n    try {\n      const packagesInfo = await B.all(B.map([\n        {\n          appPath: apkPath,\n          appId: SERVER_PACKAGE_ID,\n        }, {\n          appPath: testApkPath,\n          appId: SERVER_TEST_PACKAGE_ID,\n        },\n      ], packageInfosMapper));\n\n      let shouldUninstallServerPackages = false;\n      let shouldInstallServerPackages = false;\n      for (const {appId, appPath} of packagesInfo) {\n        if (appId === SERVER_TEST_PACKAGE_ID) {\n          const isAppInstalled = await this.adb.isAppInstalled(appId);\n\n          // There is no point in getting the state for test server,\n          // since it does not contain version info\n          if (!await this.adb.checkApkCert(appPath, appId)) {\n            await helpers.signApp(this.adb, appPath);\n            shouldUninstallServerPackages = shouldUninstallServerPackages || isAppInstalled;\n            shouldInstallServerPackages = true;\n          }\n\n          if (!isAppInstalled) {\n            shouldInstallServerPackages = true;\n          }\n          continue;\n        }\n\n        const appState = await this.adb.getApplicationInstallState(appPath, appId);\n        log.debug(`${appId} installation state: ${appState}`);\n        if (await this.adb.checkApkCert(appPath, appId)) {\n          shouldUninstallServerPackages = shouldUninstallServerPackages || [\n            this.adb.APP_INSTALL_STATE.OLDER_VERSION_INSTALLED,\n            this.adb.APP_INSTALL_STATE.NEWER_VERSION_INSTALLED,\n          ].includes(appState);\n        } else {\n          await helpers.signApp(this.adb, appPath);\n          shouldUninstallServerPackages = shouldUninstallServerPackages || ![\n            this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n          ].includes(appState);\n        }\n        shouldInstallServerPackages = shouldInstallServerPackages || shouldUninstallServerPackages || [\n          this.adb.APP_INSTALL_STATE.NOT_INSTALLED,\n        ].includes(appState);\n      }\n      log.info(`Server packages are ${shouldInstallServerPackages ? '' : 'not '}going to be (re)installed`);\n      if (shouldInstallServerPackages && shouldUninstallServerPackages) {\n        log.info('Full packages reinstall is going to be performed');\n      }\n      for (const {appId, appPath} of packagesInfo) {\n        if (shouldUninstallServerPackages) {\n          try {\n            await this.adb.uninstallApk(appId);\n          } catch (err) {\n            log.warn(`Error uninstalling '${appId}': ${err.message}`);\n          }\n        }\n        if (shouldInstallServerPackages) {\n          await this.adb.install(appPath, {\n            noIncremental: true,\n            replace: true,\n            timeout: installTimeout,\n            timeoutCapName: 'uiautomator2ServerInstallTimeout'\n          });\n        }\n      }\n    } finally {\n      await fs.rimraf(tmpRoot);\n    }\n\n    await this.verifyServicesAvailability();\n  }\n\n  async verifyServicesAvailability () {\n    log.debug(`Waiting up to ${SERVICES_LAUNCH_TIMEOUT}ms for services to be available`);\n    let isPmServiceAvailable = false;\n    let pmOutput = '';\n    let pmError = null;\n    try {\n      await waitForCondition(async () => {\n        if (!isPmServiceAvailable) {\n          pmError = null;\n          pmOutput = '';\n          try {\n            pmOutput = await this.adb.shell(['pm', 'list', 'instrumentation']);\n          } catch (e) {\n            pmError = e;\n          }\n          if (pmOutput.includes('Could not access the Package Manager')) {\n            pmError = new Error(`Problem running Package Manager: ${pmOutput}`);\n            pmOutput = ''; // remove output, so it is not printed below\n          } else if (pmOutput.includes(INSTRUMENTATION_TARGET)) {\n            pmOutput = ''; // remove output, so it is not printed below\n            log.debug(`Instrumentation target '${INSTRUMENTATION_TARGET}' is available`);\n            // eslint-disable-next-line require-atomic-updates\n            isPmServiceAvailable = true;\n          } else if (!pmError) {\n            pmError = new Error('The instrumentation target is not listed by Package Manager');\n          }\n        }\n        return isPmServiceAvailable;\n      }, {\n        waitMs: SERVICES_LAUNCH_TIMEOUT,\n        intervalMs: 1000,\n      });\n    } catch (err) {\n      log.error(`Unable to find instrumentation target '${INSTRUMENTATION_TARGET}': ${(pmError || {}).message}`);\n      if (pmOutput) {\n        log.debug('Available targets:');\n        for (const line of pmOutput.split('\\n')) {\n          log.debug(`    ${line.replace('instrumentation:', '')}`);\n        }\n      }\n    }\n  }\n\n  async startSession (caps) {\n    await this.cleanupAutomationLeftovers();\n    if (caps.skipServerInstallation) {\n      log.info(`'skipServerInstallation' is set. Attempting to use UIAutomator2 server from the device`);\n    } else {\n      log.info(`Starting UIAutomator2 server ${serverVersion}`);\n      log.info(`Using UIAutomator2 server from '${apkPath}' and test from '${testApkPath}'`);\n    }\n\n    const timeout = caps.uiautomator2ServerLaunchTimeout || SERVER_LAUNCH_TIMEOUT;\n    const timer = new timing.Timer().start();\n    let retries = 0;\n    const maxRetries = 2;\n    const delayBetweenRetries = 3000;\n    while (retries < maxRetries) {\n      log.info(`Waiting up to ${timeout}ms for UiAutomator2 to be online...`);\n      this.jwproxy.didInstrumentationExit = false;\n      await this.startInstrumentationProcess();\n      if (!this.jwproxy.didInstrumentationExit) {\n        try {\n          await waitForCondition(async () => {\n            try {\n              await this.jwproxy.command('/status', 'GET');\n              return true;\n            } catch (err) {\n              // short circuit to retry or fail fast\n              return this.jwproxy.didInstrumentationExit;\n            }\n          }, {\n            waitMs: timeout,\n            intervalMs: 1000,\n          });\n        } catch (err) {\n          log.errorAndThrow(`The instrumentation process cannot be initialized within ${timeout}ms timeout. `\n            + 'Make sure the application under test does not crash and investigate the logcat output. '\n            + `You could also try to increase the value of 'uiautomator2ServerLaunchTimeout' capability`);\n        }\n      }\n      if (!this.jwproxy.didInstrumentationExit) {\n        break;\n      }\n\n      retries++;\n      if (retries >= maxRetries) {\n        log.errorAndThrow('The instrumentation process cannot be initialized. '\n          + 'Make sure the application under test does not crash and investigate the logcat output.');\n      }\n      log.warn(`The instrumentation process has been unexpectedly terminated. `\n        + `Retrying UiAutomator2 startup (#${retries} of ${maxRetries - 1})`);\n      await this.cleanupAutomationLeftovers(true);\n      await B.delay(delayBetweenRetries);\n    }\n\n    log.debug(`The initialization of the instrumentation process took `\n      + `${timer.getDuration().asMilliSeconds.toFixed(0)}ms`);\n    await this.jwproxy.command('/session', 'POST', {\n      capabilities: {\n        firstMatch: [caps],\n        alwaysMatch: {},\n      }\n    });\n  }\n\n  async startInstrumentationProcess () {\n    const cmd = ['am', 'instrument', '-w'];\n    if (this.disableWindowAnimation) {\n      cmd.push('--no-window-animation');\n    }\n    if (_.isBoolean(this.disableSuppressAccessibilityService)) {\n      cmd.push('-e', 'DISABLE_SUPPRESS_ACCESSIBILITY_SERVICES', this.disableSuppressAccessibilityService);\n    }\n    // Disable Google analytics to prevent possible fatal exception\n    cmd.push('-e', 'disableAnalytics', true);\n    cmd.push(INSTRUMENTATION_TARGET);\n    const instrumentationProcess = this.adb.createSubProcess(['shell', ...cmd]);\n    instrumentationProcess.on('output', (stdout, stderr) => {\n      const output = _.trim(stdout || stderr);\n      if (output) {\n        instrumentationLogger.debug(output);\n      }\n    });\n    instrumentationProcess.on('exit', (code) => {\n      instrumentationLogger.debug(`The process has exited with code ${code}`);\n      this.jwproxy.didInstrumentationExit = true;\n    });\n    await instrumentationProcess.start(0);\n  }\n\n  async deleteSession () {\n    log.debug('Deleting UiAutomator2 server session');\n    // rely on jwproxy's intelligence to know what we're talking about and\n    // delete the current session\n    try {\n      await this.jwproxy.command('/', 'DELETE');\n    } catch (err) {\n      log.warn(`Did not get confirmation UiAutomator2 deleteSession worked; ` +\n          `Error was: ${err}`);\n    }\n  }\n\n  async cleanupAutomationLeftovers (strictCleanup = false) {\n    log.debug(`Performing ${strictCleanup ? 'strict' : 'shallow'} cleanup of automation leftovers`);\n\n    try {\n      const {value} = (await axios({\n        url: `http://${this.host}:${this.systemPort}/wd/hub/sessions`,\n        timeout: 500,\n      })).data;\n      const activeSessionIds = value.map(({id}) => id).filter(Boolean);\n      if (activeSessionIds.length) {\n        log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);\n        log.debug(`Cleaning up ${util.pluralize('obsolete session', activeSessionIds.length, true)}`);\n        await B.all(activeSessionIds\n          .map((id) => axios.delete(`http://${this.host}:${this.systemPort}/wd/hub/session/${id}`))\n        );\n        // Let all sessions to be properly terminated before continuing\n        await B.delay(1000);\n      } else {\n        log.debug('No obsolete sessions have been detected');\n      }\n    } catch (e) {\n      log.debug(`No obsolete sessions have been detected (${e.message})`);\n    }\n\n    try {\n      await this.adb.forceStop(SERVER_TEST_PACKAGE_ID);\n    } catch (ignore) {}\n    if (!strictCleanup) {\n      return;\n    }\n    // https://github.com/appium/appium/issues/10749\n    try {\n      await this.adb.killProcessesByName('uiautomator');\n    } catch (ignore) {}\n  }\n}\n\nexport { UiAutomator2Server, INSTRUMENTATION_TARGET, SERVER_PACKAGE_ID, SERVER_TEST_PACKAGE_ID };\nexport default UiAutomator2Server;\n"],"file":"lib/uiautomator2.js","sourceRoot":"../.."}
@@ -364,10 +364,8 @@ commands.mobileScroll = async function (opts = {}) {
364
364
  throw new errors.InvalidArgumentError(`Both strategy and selector arguments must be provided`);
365
365
  }
366
366
  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {
367
- params: {
368
- origin: toOrigin(elementId || element),
369
- strategy, selector, maxSwipes
370
- },
367
+ origin: toOrigin(elementId || element),
368
+ params: {strategy, selector, maxSwipes},
371
369
  });
372
370
  };
373
371
 
@@ -2,8 +2,14 @@ import _ from 'lodash';
2
2
  import { JWProxy, errors } from 'appium-base-driver';
3
3
  import { waitForCondition } from 'asyncbox';
4
4
  import log from './logger';
5
- import { SERVER_APK_PATH as apkPath, TEST_APK_PATH as testApkPath, version as serverVersion } from 'appium-uiautomator2-server';
6
- import { util, logger, tempDir, fs, timing } from 'appium-support';
5
+ import {
6
+ SERVER_APK_PATH as apkPath,
7
+ TEST_APK_PATH as testApkPath,
8
+ version as serverVersion
9
+ } from 'appium-uiautomator2-server';
10
+ import {
11
+ util, logger, tempDir, fs, timing
12
+ } from 'appium-support';
7
13
  import B from 'bluebird';
8
14
  import helpers from './helpers';
9
15
  import axios from 'axios';
@@ -302,13 +308,13 @@ class UiAutomator2Server {
302
308
  url: `http://${this.host}:${this.systemPort}/wd/hub/sessions`,
303
309
  timeout: 500,
304
310
  })).data;
305
- const activeSessionIds = value.map((sess) => sess.id);
311
+ const activeSessionIds = value.map(({id}) => id).filter(Boolean);
306
312
  if (activeSessionIds.length) {
307
313
  log.debug(`The following obsolete sessions are still running: ${JSON.stringify(activeSessionIds)}`);
308
- log.debug('Cleaning up the obsolete sessions');
309
- await B.all(activeSessionIds.map((id) =>
310
- axios.delete(`http://${this.host}:${this.systemPort}/wd/hub/session/${id}`)
311
- ));
314
+ log.debug(`Cleaning up ${util.pluralize('obsolete session', activeSessionIds.length, true)}`);
315
+ await B.all(activeSessionIds
316
+ .map((id) => axios.delete(`http://${this.host}:${this.systemPort}/wd/hub/session/${id}`))
317
+ );
312
318
  // Let all sessions to be properly terminated before continuing
313
319
  await B.delay(1000);
314
320
  } else {
package/package.json CHANGED
@@ -7,7 +7,7 @@
7
7
  "automated testing",
8
8
  "android"
9
9
  ],
10
- "version": "1.66.1",
10
+ "version": "1.66.2",
11
11
  "author": "appium",
12
12
  "license": "Apache-2.0",
13
13
  "repository": {
@@ -46,7 +46,7 @@
46
46
  "appium-base-driver": "^7.0.0",
47
47
  "appium-chromedriver": "^4.23.1",
48
48
  "appium-support": "^2.49.0",
49
- "appium-uiautomator2-server": "^4.22.1",
49
+ "appium-uiautomator2-server": "^4.23.0",
50
50
  "asyncbox": "^2.3.1",
51
51
  "axios": "^0.x",
52
52
  "bluebird": "^3.5.1",
@@ -89,13 +89,13 @@
89
89
  "eslint-config-appium": "^4.0.1",
90
90
  "gps-demo-app": "^2.1.1",
91
91
  "gulp": "^4.0.0",
92
- "mocha": "^8.2.1",
92
+ "mocha": "^9.0.0",
93
93
  "mocha-junit-reporter": "^2.0.0",
94
94
  "mocha-multi-reporters": "^1.1.7",
95
95
  "pngjs": "^6.0.0",
96
96
  "pre-commit": "^1.2.2",
97
97
  "rimraf": "^3.0.0",
98
- "sinon": "^10.0.0",
98
+ "sinon": "^11.0.0",
99
99
  "unzipper": "^0.10.0",
100
100
  "wd": "^1.10.3",
101
101
  "xmldom": "^0.x",