to
Appium's
Contributing
Open Source Ecosystem
data:image/s3,"s3://crabby-images/163a1/163a19845ff0f5336ede0c180305d1753c3d9b96" alt=""
data:image/s3,"s3://crabby-images/a432c/a432c1a5000e5bd1f370b57867380c26723f3033" alt=""
I'm Srinivasan
Lead Consultant
@srinivasanskr
srinivasanTarget
data:image/s3,"s3://crabby-images/cf763/cf763ebaefb7da530bfa9ad7c14b745f4f45d336" alt=""
data:image/s3,"s3://crabby-images/527b9/527b9161cf7f8edf7c8a0c8e51bab1426cc1a7bf" alt=""
Hello,
data:image/s3,"s3://crabby-images/a4261/a4261f6d0e28bd9e782f557352d3d0e137817fa0" alt=""
data:image/s3,"s3://crabby-images/7bc0c/7bc0caa20bcc8c21d052753013a9e047c8b8bcc1" alt=""
Problem Statement
Timed out waiting for Espresso Server to start due to Socket exception.
Android Architecture
data:image/s3,"s3://crabby-images/db079/db079280e99f22877e5c332107944cdc85b1425b" alt=""
data:image/s3,"s3://crabby-images/e0a9a/e0a9af540652abaddbca241ab913f9ad2052b2e4" alt=""
data:image/s3,"s3://crabby-images/ce199/ce1999599ee582f6fda37b9a215adcf77fc2e1b7" alt=""
data:image/s3,"s3://crabby-images/9bc61/9bc61900a687ac966438f96790f4adef1afb8252" alt=""
W3C
Server Request
data:image/s3,"s3://crabby-images/32991/32991391e3c9dfcb38d9ff9caaaa339515544367" alt=""
Server
Client
Appium Server
Driver(espresso)
data:image/s3,"s3://crabby-images/03e62/03e6260123e487cec0a38094090d6cb27c9bfbff" alt=""
data:image/s3,"s3://crabby-images/03e62/03e6260123e487cec0a38094090d6cb27c9bfbff" alt=""
data:image/s3,"s3://crabby-images/a432c/a432c1a5000e5bd1f370b57867380c26723f3033" alt=""
data:image/s3,"s3://crabby-images/05d67/05d67ac36fa98e78b8b6deddc0b9b70ba1f15472" alt=""
Server Request
Plugins
Appium Android Driver
Appium ADB
Appium Espresso Driver
Appium Base Driver
Appium UIA2 Driver
Appium Support
data:image/s3,"s3://crabby-images/a432c/a432c1a5000e5bd1f370b57867380c26723f3033" alt=""
data:image/s3,"s3://crabby-images/9e3cd/9e3cd9f5bf4e81bb22e79bed6996e561e3afc8dd" alt=""
Initialisation
Creation
Instrumentation
Execution
Android Life Cycle
DesiredCapabilities capabilities = new DesiredCapabilities();
capabilities.setCapability("deviceName", "Android Emulator");
capabilities.setCapability("app", app.getAbsolutePath());
capabilities.setCapability("appPackage", "io.appium.android.apis");
capabilities.setCapability("appActivity", ".ApiDemos");
capabilities.setCapability("automationName", "Espresso"); driver = new AndroidDriver<WebElement>(getServiceUrl(), capabilities);
Intialization
Driver Creation
App Launch State
automationNameCap = automationNameCap.toLowerCase();
try {
const {driverPackage, driverClassName} = DRIVER_MAP[automationNameCap];
const driver = require('appium-espresso-driver')[driverClassName];
return {
driver,
version: this.getDriverVersion(driver.name, driverPackage),
};
} catch () {
Appium Base Driver
Protocol Proxy
UIA2 or
Espresso Driver
Espresso Driver
- EspressoDriver extends BaseDriver
-
Merges the default values with desired capability
async createSession (...args) {
try {
let [sessionId, caps] = await super.createSession(...args);
let serverDetails = {
platform: 'LINUX',
webStorageEnabled: false,
takesScreenshot: true,
javascriptEnabled: true,
databaseEnabled: false,
networkConnectionEnabled: true,
locationContextEnabled: false,
warnings: {},
desired: Object.assign({}, this.caps)
};
this.caps = Object.assign(serverDetails, this.caps);
Espresso Driver
async startEspressoSession () {
logger.info(`EspressoDriver version: ${version}`);
await helpers.getDeviceInfoFromCaps(this.opts);
....
await androidHelpers.createADB(this.opts);
....
this.initEspressoServer();
appium-android-driver
appium-adb
initEspressoServer () {
this.espresso = new EspressoRunner({
host: this.opts.remoteAdbHost || this.opts.host || 'localhost',
systemPort: this.opts.systemPort,
devicePort: DEVICE_PORT,
adb: this.adb,
apk: this.opts.app,
tmpDir: this.opts.tmpDir,
appPackage: this.opts.appPackage,
appActivity: this.opts.appActivity,
forceEspressoRebuild: !!this.opts.forceEspressoRebuild,
serverLaunchTimeout: this.opts.espressoServerLaunchTimeout,
androidInstallTimeout: this.opts.androidInstallTimeout,
});
this.proxyReqRes = this.espresso.proxyReqRes.bind(this.espresso);
}
espresso-driver/lib/driver.js
Espresso Session Creation
adb forward tcp:8300 tcp:6791
AUT
await this.adb.forwardPort(this.opts.systemPort, DEVICE_PORT);
await this.adb.install(this.modServerPath, { replace: false,timeout: this.androidInstallTimeout });
Instrumentation
data:image/s3,"s3://crabby-images/5bda3/5bda352026065ed50ff0c45956122a618785b467" alt=""
data:image/s3,"s3://crabby-images/5bda3/5bda352026065ed50ff0c45956122a618785b467" alt=""
data:image/s3,"s3://crabby-images/042aa/042aa2afde6b9029a9b2dde51f9c42d9e56be790" alt=""
async startSession (caps) {
await this.cleanupSessionLeftovers();
const cmd = [
'shell',
'am', 'instrument',
'-w',
'-e', 'debug', process.env.ESPRESSO_JAVA_DEBUG === 'true' ? 'true' : 'false',
`${TEST_APK_PKG}/androidx.test.runner.AndroidJUnitRunner`,
];
logger.info(`Starting Espresso Server v${version} with cmd: adb ${cmd.join(' ')}`);
Execution
try {
await retryInterval(20, 1000, async () => {
await this.jwproxy.command('/status', 'GET');
});
} catch (e) {
if (hasSocketError) {
logger.errorAndThrow('....');
} else {
logger.errorAndThrow(`Timed out waiting for Espresso Server to start. Original error: ${e.message}`);
}
}
await this.jwproxy.command('/session', 'POST', {desiredCapabilities: caps});
Execution
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
{
"desiredCapabilities": {
"desired": {
"platformName": "android",
"app": "/Users/saikrisv/git/VodQaAdvancedAppium/VodQA.apk",
"automationName": "Espresso",
"deviceName": "Android Emulator",
"launchTimeout": 900000
},
"platformName": "android",
"app": "/Users/saikrisv/git/VodQaAdvancedAppium/VodQA.apk",
"automationName": "Espresso",
"deviceName": "emulator-5554",
"launchTimeout": 900000,
"deviceUDID": "emulator-5554",
"appPackage": "com.vodqareactnative",
"appWaitPackage": "com.vodqareactnative",
"appActivity": "com.vodqareactnative.MainActivity",
"appWaitActivity": "com.vodqareactnative.MainActivity"
}
Execution
POST /wd/hub/session
data:image/s3,"s3://crabby-images/03cc2/03cc26a93028cb0dae9c6ce7de8cad40efaf12b9" alt=""
Demo
IOS Session Creation
Problem Statement
Unable to launch WebDriverAgent because of xcodebuild failure: "xcodebuild failed with code 65".
data:image/s3,"s3://crabby-images/db079/db079280e99f22877e5c332107944cdc85b1425b" alt=""
data:image/s3,"s3://crabby-images/e0a9a/e0a9af540652abaddbca241ab913f9ad2052b2e4" alt=""
data:image/s3,"s3://crabby-images/ce199/ce1999599ee582f6fda37b9a215adcf77fc2e1b7" alt=""
data:image/s3,"s3://crabby-images/9bc61/9bc61900a687ac966438f96790f4adef1afb8252" alt=""
W3C
Server Request
data:image/s3,"s3://crabby-images/32991/32991391e3c9dfcb38d9ff9caaaa339515544367" alt=""
WDA
Appium XCUI driver
data:image/s3,"s3://crabby-images/549de/549de3b81c122d8fc1457c906e4e51e97f2d0eb9" alt=""
data:image/s3,"s3://crabby-images/7cd20/7cd207f4e384e3296ddc49119869e70f3b7aabef" alt=""
IOS Architecture
Appium Server
Client
data:image/s3,"s3://crabby-images/a432c/a432c1a5000e5bd1f370b57867380c26723f3033" alt=""
appium-ios-simulator
node-simctl
appium-xcode
appium-remote-debugger
Appium XCUI Driver
Appium IOS Driver
Appium Base Driver
data:image/s3,"s3://crabby-images/a432c/a432c1a5000e5bd1f370b57867380c26723f3033" alt=""
data:image/s3,"s3://crabby-images/9e3cd/9e3cd9f5bf4e81bb22e79bed6996e561e3afc8dd" alt=""
Intialization
APP & WDA
Execution
iOS Life Cycle
Creation
DesiredCapabilities caps = new DesiredCapabilities();
caps.setCapability("platformName", "iOS");
caps.setCapability("platformVersion", "12.2");
caps.setCapability("deviceName", "iPhone Xs");
caps.setCapability(IOSMobileCapabilityType.BUNDLE_ID, BUNDLE_ID);
caps.setCapability("udid", "666B367A-62E7-45F1-B430-36C2A1971BB8");
driver = new IOSDriver<MobileElement>(new URL("http://localhost:4723/wd/hub"), caps);
Intialization
Driver Creation
App Launch State
Protocol Proxy
XCUI Driver
Appium Base Driver
let automationNameCap = caps.automationName;
automationNameCap = automationNameCap.toLowerCase();
try {
const {driverPackage, driverClassName} = DRIVER_MAP[automationNameCap];
return {
driver,
version: this.getDriverVersion(driver.name, driverPackage),
};
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
const driver = require['appium-xcuitest-driver'];
- XCUIDriver extends BaseDriver
-
Merges the default values to the user given capability
../appium-xcuitest-driver/lib/driver.js
XCUI Driver
// merge server capabilities + desired capabilities
caps = Object.assign({}, defaultServerCaps, caps);
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
async createSession (...args) {
this.lifecycleData = {};
try {
let [sessionId, caps] = await super.createSession(...args);
this.opts.sessionId = sessionId;
await this.start();
../appium-xcuitest-driver/lib/simulator-management.js
const {device, udid, realDevice} = await this.determineDevice();
return await getSimulator(appiumTestDevice.udid);
Device Name Platform Version
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
node-simctl
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
appium-ios-simulator
async function getExistingSim (opts) {
const devices = await getDevices(opts.platformVersion);
Simulators
Simulators/Real Device
log.debug(`Available devices: ${devices.join(', ')}`); if (!devices.includes(this.opts.udid)) { if (await simExists(this.opts.udid)) { return {device, realDevice: false, udid: this.opts.udid}; } throw new Error(`Unknown device or simulator UDID: '${this.opts.udid}'`); } } if (_.isEmpty(this.opts.platformVersion)) { log.info('Trying to determine platformVersion from ideviceinfo output'); try { const {stdout} = await exec('ideviceinfo', [ '-u', this.opts.udid, '-s', '-k', 'ProductVersion', ]); this.opts.platformVersion = util.coerceVersion(stdout.trim(), false); } catch (e) { log.warn(`Cannot determine real device platform version. Original error: ${e.message}`); } } return {device, realDevice: true, udid: this.opts.udid}; }
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
node-simctl
idevice_id
const device = await getSimulator(this.opts.udid);
const device = await getRealDeviceObj(this.opts.udid);
const devices = await getConnectedDevices();
App Installation
async function installToSimulator (device, app, bundleId, noReset = true) {
if (bundleId) {
if (await device.isAppInstalled(bundleId)) {
if (noReset) {
log.debug(`App '${bundleId}' is already installed. No need to reinstall.`);
return;
}
log.debug(`Reset requested. Removing app with id '${bundleId}' from the device`);
await device.removeApp(bundleId);
}
}
......
try {
} catch (e) {
}
log.info('Retrying application install');
....
}
log.debug('The app has been installed successfully.');
} finally {
if (tmpRoot && await fs.exists(tmpRoot)) {
await fs.rimraf(tmpRoot);
}
}
}
node-simctl
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
await device.installApp(app);
App Installation
ios-deploy
async function installToRealDevice (device, app, bundleId, noReset = true) { if (!device.udid || !app) { log.debug('No device id or app, not installing to real device.'); return; } if (await device.isAppInstalled(bundleId)) { if (noReset) { log.debug(`App '${bundleId}' is already installed. No need to reinstall.`); return; } log.debug(`Reset requested. Removing app with id '${bundleId}' from the device`); await device.remove(bundleId); } }
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
await device.install(app);
WDA Start
wdaLocalPort
data:image/s3,"s3://crabby-images/5d2c7/5d2c74db6555a52f905fda0c5a6ae3222ac16a5b" alt=""
async start (buildOnly = false) {
Execution
this.xcodebuild.processExited = true;
if (this.xcodebuild._wda_error_occurred || (!signal && code !== 0)) {
return reject(new Error(`xcodebuild failed with code ${code}${EOL}` +
`xcodebuild error message:${EOL}${this.xcodebuild._wda_error_message}`));
}
this.xcodebuild = await this.createSubProcess(buildOnly);
this.xcodebuild._wda_error_message = '';
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
try {
await retryInterval(15, 1000, async () => {
try {
this.proxyCommand('/status', 'GET');
await this.startWdaSession(this.opts.bundleId, this.opts.processArguments);
} catch (err) {
originalStacktrace = err.stack;
log.debug(`Failed to create WDA session (${err.message}). Retrying...`);
throw err;
}
});
}
data:image/s3,"s3://crabby-images/cd912/cd912ee9f83af37d5d82c24a3a8b3a6868a4b5a1" alt=""
Execution
FindElement
driver.findElement(By.accessibility)
[HTTP] --> POST /wd/hub/session/ca5d6df3-c557-40ae-973a-6620d4a9fc42/elements
[HTTP] {"using":"accessibility id","value":"login"}
data:image/s3,"s3://crabby-images/14cf5/14cf5422a20e8d79b29725b74ffb0c11969dc41b" alt=""
@saikrisv
saikrisv@thoughtworks.com
github.com/saikrishna321
data:image/s3,"s3://crabby-images/7bc0c/7bc0caa20bcc8c21d052753013a9e047c8b8bcc1" alt=""
data:image/s3,"s3://crabby-images/a4261/a4261f6d0e28bd9e782f557352d3d0e137817fa0" alt=""
data:image/s3,"s3://crabby-images/7a745/7a7450253a89a59c89f0a7a3173551a43e0d39a4" alt=""
@srinivasanskr
sekars@thoughtworks.com
github.com/srinivasantarget
data:image/s3,"s3://crabby-images/7bc0c/7bc0caa20bcc8c21d052753013a9e047c8b8bcc1" alt=""
data:image/s3,"s3://crabby-images/a4261/a4261f6d0e28bd9e782f557352d3d0e137817fa0" alt=""
data:image/s3,"s3://crabby-images/7a745/7a7450253a89a59c89f0a7a3173551a43e0d39a4" alt=""
data:image/s3,"s3://crabby-images/73876/73876083f247d40b3ce0cfc10ef2b3f5106cfce8" alt=""
data:image/s3,"s3://crabby-images/1f9d4/1f9d40bb2ef7d724f121dca890ae991b857981a3" alt=""
data:image/s3,"s3://crabby-images/3b070/3b07072b120188f4ab76030e077654af5866c849" alt=""
data:image/s3,"s3://crabby-images/248ad/248ad4014087abfa2eeec2427fe4685847374524" alt=""
Contributing to Appium's Open Source Ecosystem
By Srinivasan Sekar
Contributing to Appium's Open Source Ecosystem
- 191