mirror of
https://github.com/appium/appium.git
synced 2026-02-20 10:20:05 -06:00
82 lines
2.9 KiB
JavaScript
82 lines
2.9 KiB
JavaScript
import {BasePlugin} from 'appium/plugin';
|
|
import {errors} from 'appium/driver';
|
|
import {transformSourceXml} from './source';
|
|
import {transformQuery} from './xpath';
|
|
import log from './logger';
|
|
|
|
export default class UniversalXMLPlugin extends BasePlugin {
|
|
async getPageSource(next, driver, sessId, addIndexPath = false) {
|
|
const source = next ? await next() : await driver.getPageSource();
|
|
const metadata = {};
|
|
const {platformName} = driver.caps;
|
|
if (platformName.toLowerCase() === 'android') {
|
|
metadata.appPackage = driver.opts.appPackage;
|
|
}
|
|
const {xml, unknowns} = await transformSourceXml(source, platformName.toLowerCase(), {
|
|
metadata,
|
|
addIndexPath,
|
|
});
|
|
if (unknowns.nodes.length) {
|
|
log.warn(
|
|
`The XML mapper found ${unknowns.nodes.length} node(s) / ` +
|
|
`tag name(s) that it didn't know about. These should be ` +
|
|
`reported to improve the quality of the plugin: ` +
|
|
unknowns.nodes.join(', '),
|
|
);
|
|
}
|
|
if (unknowns.attrs.length) {
|
|
log.warn(
|
|
`The XML mapper found ${unknowns.attrs.length} attributes ` +
|
|
`that it didn't know about. These should be reported to ` +
|
|
`improve the quality of the plugin: ` +
|
|
unknowns.attrs.join(', '),
|
|
);
|
|
}
|
|
return xml;
|
|
}
|
|
|
|
async findElement(...args) {
|
|
return await this._find(false, ...args);
|
|
}
|
|
|
|
async findElements(...args) {
|
|
return await this._find(true, ...args);
|
|
}
|
|
|
|
async _find(multiple, next, driver, strategy, selector) {
|
|
const {platformName} = driver.caps;
|
|
if (strategy.toLowerCase() !== 'xpath' || (await driver.getCurrentContext()) !== 'NATIVE_APP') {
|
|
return await next();
|
|
}
|
|
const xml = await this.getPageSource(null, driver, null, true);
|
|
let newSelector = transformQuery(selector, xml, multiple);
|
|
|
|
// if the selector was not able to be transformed, that means no elements were found that
|
|
// matched, so do the appropriate thing based on element vs elements
|
|
if (newSelector === null) {
|
|
log.warn(
|
|
`Selector was not able to be translated to underlying XML. Either the requested ` +
|
|
`element does not exist or there was an error in translation`,
|
|
);
|
|
if (multiple) {
|
|
return [];
|
|
}
|
|
throw new errors.NoSuchElementError();
|
|
}
|
|
|
|
if (platformName.toLowerCase() === 'ios') {
|
|
// with the XCUITest driver, the <AppiumAUT> wrapper element is present in the source but is
|
|
// not present in the source considered by WDA, so our index path based xpath queries will
|
|
// not work with WDA as-is. We need to remove the first path segment.
|
|
newSelector = newSelector.replace(/^\/\*\[1\]/, '');
|
|
}
|
|
log.info(`Selector was translated to: ${newSelector}`);
|
|
|
|
// otherwise just run the transformed query!
|
|
const finder = multiple ? 'findElements' : 'findElement';
|
|
return await driver[finder](strategy, newSelector);
|
|
}
|
|
}
|
|
|
|
export {UniversalXMLPlugin};
|