appium-espresso-driver 2.0.1 → 2.1.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.
- package/README.md +2 -2
- package/build/lib/driver.js +123 -35
- package/build/lib/server-builder.js +3 -2
- package/espresso-server/app/build/outputs/apk/androidTest/debug/app-debug-androidTest.apk +0 -0
- package/espresso-server/app/build/outputs/apk/androidTest/debug/output-metadata.json +5 -3
- package/espresso-server/app/build.gradle.kts +18 -18
- package/espresso-server/app/src/test/java/io/appium/espressoserver/test/helpers/KReflectionUtilsTest.kt +4 -4
- package/espresso-server/app/src/test/java/io/appium/espressoserver/test/model/MobileClickTest.kt +2 -2
- package/espresso-server/app/src/test/java/io/appium/espressoserver/test/model/MobileSwipeTest.kt +3 -3
- package/espresso-server/build.gradle.kts +2 -1
- package/espresso-server/gradle/wrapper/gradle-wrapper.properties +1 -1
- package/lib/driver.js +132 -37
- package/lib/server-builder.js +2 -1
- package/npm-shrinkwrap.json +431 -586
- package/package.json +3 -3
|
@@ -171,7 +171,8 @@ class ServerBuilder {
|
|
|
171
171
|
|
|
172
172
|
const gradlebuild = new _teen_process.SubProcess(cmd, args, {
|
|
173
173
|
cwd: this.serverPath,
|
|
174
|
-
stdio: ['ignore', 'pipe', 'pipe']
|
|
174
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
175
|
+
windowsVerbatimArguments: true
|
|
175
176
|
});
|
|
176
177
|
let buildLastLines = [];
|
|
177
178
|
const logMsg = `Output from Gradle ${this.showGradleLog ? 'will' : 'will not'} be logged`;
|
|
@@ -211,4 +212,4 @@ var _default = ServerBuilder;
|
|
|
211
212
|
exports.default = _default;require('source-map-support').install();
|
|
212
213
|
|
|
213
214
|
|
|
214
|
-
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/server-builder.js"],"names":["GRADLE_VERSION_KEY","GRADLE_URL_PREFIX","GRADLE_URL_TEMPLATE","GRADLE_MAX_ERROR_LOG_LINES","DEPENDENCY_PROP_NAMES","VERSION_KEYS","gradleLog","logger","getLogger","buildServerSigningConfig","args","zipAlign","keystoreFile","keystorePassword","keyAlias","keyPassword","ServerBuilder","constructor","serverPath","showGradleLog","buildConfiguration","versionConfiguration","toolsVersions","serverVersions","_","reduce","acc","value","key","includes","log","warn","testAppPackage","signingConfig","propName","build","setGradleWrapperVersion","insertAdditionalDependencies","runBuildProcess","getCommand","cmd","system","isWindows","buildProperty","filter","map","serverVersion","gradleProperty","charAt","toUpperCase","slice","Boolean","push","keys","upperFirst","k","v","version","propertiesPath","path","resolve","originalProperties","fs","readFile","newProperties","updateGradleDistUrl","writeFile","propertiesContent","replace","RegExp","escapeRegExp","hasAdditionalDeps","isArray","Error","isEmpty","line","trim","dep","test","buildPath","configuration","prefix","deps","info","debug","join","gradlebuild","SubProcess","cwd","stdio","buildLastLines","logMsg","on","startsWith","EOL","length","start","err","msg","message","errorAndThrow"],"mappings":"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","sourcesContent":["import { SubProcess } from 'teen_process';\nimport { fs, logger, system } from '@appium/support';\nimport _ from 'lodash';\nimport log from './logger';\nimport path from 'path';\nimport { EOL } from 'os';\nimport { updateDependencyLines } from './utils';\n\nconst GRADLE_VERSION_KEY = 'gradle';\nconst GRADLE_URL_PREFIX = 'distributionUrl=';\nconst GRADLE_URL_TEMPLATE = 'https\\\\://services.gradle.org/distributions/gradle-VERSION-all.zip';\nconst GRADLE_MAX_ERROR_LOG_LINES = 15;\nconst DEPENDENCY_PROP_NAMES = ['additionalAppDependencies', 'additionalAndroidTestDependencies'];\n\nconst VERSION_KEYS = [\n  GRADLE_VERSION_KEY,\n  'androidGradlePlugin',\n  'compileSdk',\n  'buildTools',\n  'minSdk',\n  'targetSdk',\n  'kotlin',\n  'sourceCompatibility',\n  'targetCompatibility',\n  'jvmTarget',\n];\n\nconst gradleLog = logger.getLogger('Gradle');\n\nfunction buildServerSigningConfig (args) {\n  return {\n    zipAlign: true,\n    keystoreFile: args.keystoreFile,\n    keystorePassword: args.keystorePassword,\n    keyAlias: args.keyAlias,\n    keyPassword: args.keyPassword\n  };\n}\n\nclass ServerBuilder {\n  constructor (args = {}) {\n    this.serverPath = args.serverPath;\n    this.showGradleLog = args.showGradleLog;\n\n    const buildConfiguration = args.buildConfiguration || {};\n\n    const versionConfiguration = buildConfiguration.toolsVersions || {};\n    this.serverVersions = _.reduce(versionConfiguration, (acc, value, key) => {\n      if (VERSION_KEYS.includes(key)) {\n        acc[key] = value;\n      } else {\n        log.warn(`Got unexpected '${key}' in toolsVersion block of the build configuration`);\n      }\n      return acc;\n    }, {});\n\n    this.testAppPackage = args.testAppPackage;\n    this.signingConfig = args.signingConfig;\n\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      this[propName] = buildConfiguration[propName] || [];\n    }\n  }\n\n  async build () {\n    if (this.serverVersions[GRADLE_VERSION_KEY]) {\n      await this.setGradleWrapperVersion(this.serverVersions[GRADLE_VERSION_KEY]);\n    }\n\n    await this.insertAdditionalDependencies();\n\n    await this.runBuildProcess();\n  }\n\n  getCommand () {\n    const cmd = system.isWindows() ? 'gradlew.bat' : './gradlew';\n    const buildProperty = (key, value) => value ? `-P${key}=${value}` : null;\n    let args = VERSION_KEYS\n      .filter((key) => key !== GRADLE_VERSION_KEY)\n      .map((key) => {\n        const serverVersion = this.serverVersions[key];\n        const gradleProperty = `appium${key.charAt(0).toUpperCase()}${key.slice(1)}`;\n        return buildProperty(gradleProperty, serverVersion);\n      })\n      .filter(Boolean);\n\n    if (this.signingConfig) {\n      args.push(...(\n        _.keys(this.signingConfig)\n        .map((key) => [`appium${_.upperFirst(key)}`, this.signingConfig[key]])\n        .map(([k, v]) => buildProperty(k, v))\n        .filter(Boolean)\n      ));\n    }\n\n    if (this.testAppPackage) {\n      args.push(buildProperty('appiumTargetPackage', this.testAppPackage));\n    }\n    args.push('app:assembleAndroidTest');\n\n    return {cmd, args};\n  }\n\n  async setGradleWrapperVersion (version) {\n    const propertiesPath = path.resolve(this.serverPath, 'gradle', 'wrapper', 'gradle-wrapper.properties');\n    const originalProperties = await fs.readFile(propertiesPath, 'utf8');\n    const newProperties = this.updateGradleDistUrl(originalProperties, version);\n    await fs.writeFile(propertiesPath, newProperties, 'utf8');\n  }\n\n  updateGradleDistUrl (propertiesContent, version) {\n    return propertiesContent.replace(\n      new RegExp(`^(${_.escapeRegExp(GRADLE_URL_PREFIX)}).+$`, 'gm'),\n      `$1${GRADLE_URL_TEMPLATE.replace('VERSION', version)}`\n    );\n  }\n\n  async insertAdditionalDependencies () {\n    let hasAdditionalDeps = false;\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      if (!_.isArray(this[propName])) {\n        throw new Error(`'${propName}' must be an array`);\n      }\n      if (_.isEmpty(this[propName].filter((line) => _.trim(line)))) {\n        continue;\n      }\n\n      for (const dep of this[propName]) {\n        if (/[\\s'\\\\$]/.test(dep)) {\n          throw new Error('Single quotes, dollar characters and whitespace characters' +\n            ` are disallowed in additional dependencies: ${dep}`);\n        }\n      }\n      hasAdditionalDeps = true;\n    }\n    if (!hasAdditionalDeps) {\n      return;\n    }\n\n    const buildPath = path.resolve(this.serverPath, 'app', 'build.gradle.kts');\n    let configuration = await fs.readFile(buildPath, 'utf8');\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      const prefix = propName === DEPENDENCY_PROP_NAMES[0]\n        ? 'implementation'\n        : 'androidTestImplementation';\n      const deps = this[propName]\n        .filter((line) => _.trim(line))\n        .map((line) => `${prefix}(\"${line}\")`);\n      if (_.isEmpty(deps)) {\n        continue;\n      }\n\n      log.info(`Adding the following ${propName} to build.gradle.kts: ${deps}`);\n      configuration = updateDependencyLines(configuration, propName, deps);\n    }\n    await fs.writeFile(buildPath, configuration, 'utf8');\n  }\n\n  async runBuildProcess () {\n    const {cmd, args} = this.getCommand();\n    log.debug(`Beginning build with command '${cmd} ${args.join(' ')}' ` +\n      `in directory '${this.serverPath}'`);\n    const gradlebuild = new SubProcess(cmd, args, {\n      cwd: this.serverPath,\n      stdio: ['ignore', 'pipe', 'pipe'],\n    });\n    let buildLastLines = [];\n\n    const logMsg = `Output from Gradle ${this.showGradleLog ? 'will' : 'will not'} be logged`;\n    log.debug(`${logMsg}. To change this, use 'showGradleLog' desired capability`);\n    gradlebuild.on('stream-line', (line) => {\n      if (this.showGradleLog) {\n        if (line.startsWith('[STDERR]')) {\n          gradleLog.warn(line);\n        } else {\n          gradleLog.info(line);\n        }\n      }\n      buildLastLines.push(`${EOL}${line}`);\n      if (buildLastLines.length > GRADLE_MAX_ERROR_LOG_LINES) {\n        buildLastLines = buildLastLines.slice(-GRADLE_MAX_ERROR_LOG_LINES);\n      }\n    });\n\n    try {\n      await gradlebuild.start();\n      await gradlebuild.join();\n    } catch (err) {\n      let msg = `Unable to build Espresso server - ${err.message}\\n` +\n        `Gradle error message:${EOL}${buildLastLines}`;\n      log.errorAndThrow(msg);\n    }\n  }\n}\n\nexport { ServerBuilder, VERSION_KEYS, GRADLE_URL_TEMPLATE, buildServerSigningConfig };\nexport default ServerBuilder;\n"],"file":"lib/server-builder.js","sourceRoot":"../.."}
|
|
215
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/server-builder.js"],"names":["GRADLE_VERSION_KEY","GRADLE_URL_PREFIX","GRADLE_URL_TEMPLATE","GRADLE_MAX_ERROR_LOG_LINES","DEPENDENCY_PROP_NAMES","VERSION_KEYS","gradleLog","logger","getLogger","buildServerSigningConfig","args","zipAlign","keystoreFile","keystorePassword","keyAlias","keyPassword","ServerBuilder","constructor","serverPath","showGradleLog","buildConfiguration","versionConfiguration","toolsVersions","serverVersions","_","reduce","acc","value","key","includes","log","warn","testAppPackage","signingConfig","propName","build","setGradleWrapperVersion","insertAdditionalDependencies","runBuildProcess","getCommand","cmd","system","isWindows","buildProperty","filter","map","serverVersion","gradleProperty","charAt","toUpperCase","slice","Boolean","push","keys","upperFirst","k","v","version","propertiesPath","path","resolve","originalProperties","fs","readFile","newProperties","updateGradleDistUrl","writeFile","propertiesContent","replace","RegExp","escapeRegExp","hasAdditionalDeps","isArray","Error","isEmpty","line","trim","dep","test","buildPath","configuration","prefix","deps","info","debug","join","gradlebuild","SubProcess","cwd","stdio","windowsVerbatimArguments","buildLastLines","logMsg","on","startsWith","EOL","length","start","err","msg","message","errorAndThrow"],"mappings":"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","sourcesContent":["import { SubProcess } from 'teen_process';\nimport { fs, logger, system } from '@appium/support';\nimport _ from 'lodash';\nimport log from './logger';\nimport path from 'path';\nimport { EOL } from 'os';\nimport { updateDependencyLines } from './utils';\n\nconst GRADLE_VERSION_KEY = 'gradle';\nconst GRADLE_URL_PREFIX = 'distributionUrl=';\nconst GRADLE_URL_TEMPLATE = 'https\\\\://services.gradle.org/distributions/gradle-VERSION-all.zip';\nconst GRADLE_MAX_ERROR_LOG_LINES = 15;\nconst DEPENDENCY_PROP_NAMES = ['additionalAppDependencies', 'additionalAndroidTestDependencies'];\n\nconst VERSION_KEYS = [\n  GRADLE_VERSION_KEY,\n  'androidGradlePlugin',\n  'compileSdk',\n  'buildTools',\n  'minSdk',\n  'targetSdk',\n  'kotlin',\n  'sourceCompatibility',\n  'targetCompatibility',\n  'jvmTarget'\n];\n\nconst gradleLog = logger.getLogger('Gradle');\n\nfunction buildServerSigningConfig (args) {\n  return {\n    zipAlign: true,\n    keystoreFile: args.keystoreFile,\n    keystorePassword: args.keystorePassword,\n    keyAlias: args.keyAlias,\n    keyPassword: args.keyPassword\n  };\n}\n\nclass ServerBuilder {\n  constructor (args = {}) {\n    this.serverPath = args.serverPath;\n    this.showGradleLog = args.showGradleLog;\n\n    const buildConfiguration = args.buildConfiguration || {};\n\n    const versionConfiguration = buildConfiguration.toolsVersions || {};\n    this.serverVersions = _.reduce(versionConfiguration, (acc, value, key) => {\n      if (VERSION_KEYS.includes(key)) {\n        acc[key] = value;\n      } else {\n        log.warn(`Got unexpected '${key}' in toolsVersion block of the build configuration`);\n      }\n      return acc;\n    }, {});\n\n    this.testAppPackage = args.testAppPackage;\n    this.signingConfig = args.signingConfig;\n\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      this[propName] = buildConfiguration[propName] || [];\n    }\n  }\n\n  async build () {\n    if (this.serverVersions[GRADLE_VERSION_KEY]) {\n      await this.setGradleWrapperVersion(this.serverVersions[GRADLE_VERSION_KEY]);\n    }\n\n    await this.insertAdditionalDependencies();\n\n    await this.runBuildProcess();\n  }\n\n  getCommand () {\n    const cmd = system.isWindows() ? 'gradlew.bat' : './gradlew';\n    const buildProperty = (key, value) => value ? `-P${key}=${value}` : null;\n    let args = VERSION_KEYS\n      .filter((key) => key !== GRADLE_VERSION_KEY)\n      .map((key) => {\n        const serverVersion = this.serverVersions[key];\n        const gradleProperty = `appium${key.charAt(0).toUpperCase()}${key.slice(1)}`;\n        return buildProperty(gradleProperty, serverVersion);\n      })\n      .filter(Boolean);\n\n    if (this.signingConfig) {\n      args.push(...(\n        _.keys(this.signingConfig)\n        .map((key) => [`appium${_.upperFirst(key)}`, this.signingConfig[key]])\n        .map(([k, v]) => buildProperty(k, v))\n        .filter(Boolean)\n      ));\n    }\n\n    if (this.testAppPackage) {\n      args.push(buildProperty('appiumTargetPackage', this.testAppPackage));\n    }\n    args.push('app:assembleAndroidTest');\n\n    return {cmd, args};\n  }\n\n  async setGradleWrapperVersion (version) {\n    const propertiesPath = path.resolve(this.serverPath, 'gradle', 'wrapper', 'gradle-wrapper.properties');\n    const originalProperties = await fs.readFile(propertiesPath, 'utf8');\n    const newProperties = this.updateGradleDistUrl(originalProperties, version);\n    await fs.writeFile(propertiesPath, newProperties, 'utf8');\n  }\n\n  updateGradleDistUrl (propertiesContent, version) {\n    return propertiesContent.replace(\n      new RegExp(`^(${_.escapeRegExp(GRADLE_URL_PREFIX)}).+$`, 'gm'),\n      `$1${GRADLE_URL_TEMPLATE.replace('VERSION', version)}`\n    );\n  }\n\n  async insertAdditionalDependencies () {\n    let hasAdditionalDeps = false;\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      if (!_.isArray(this[propName])) {\n        throw new Error(`'${propName}' must be an array`);\n      }\n      if (_.isEmpty(this[propName].filter((line) => _.trim(line)))) {\n        continue;\n      }\n\n      for (const dep of this[propName]) {\n        if (/[\\s'\\\\$]/.test(dep)) {\n          throw new Error('Single quotes, dollar characters and whitespace characters' +\n            ` are disallowed in additional dependencies: ${dep}`);\n        }\n      }\n      hasAdditionalDeps = true;\n    }\n    if (!hasAdditionalDeps) {\n      return;\n    }\n\n    const buildPath = path.resolve(this.serverPath, 'app', 'build.gradle.kts');\n    let configuration = await fs.readFile(buildPath, 'utf8');\n    for (const propName of DEPENDENCY_PROP_NAMES) {\n      const prefix = propName === DEPENDENCY_PROP_NAMES[0]\n        ? 'implementation'\n        : 'androidTestImplementation';\n      const deps = this[propName]\n        .filter((line) => _.trim(line))\n        .map((line) => `${prefix}(\"${line}\")`);\n      if (_.isEmpty(deps)) {\n        continue;\n      }\n\n      log.info(`Adding the following ${propName} to build.gradle.kts: ${deps}`);\n      configuration = updateDependencyLines(configuration, propName, deps);\n    }\n    await fs.writeFile(buildPath, configuration, 'utf8');\n  }\n\n  async runBuildProcess () {\n    const {cmd, args} = this.getCommand();\n    log.debug(`Beginning build with command '${cmd} ${args.join(' ')}' ` +\n      `in directory '${this.serverPath}'`);\n    const gradlebuild = new SubProcess(cmd, args, {\n      cwd: this.serverPath,\n      stdio: ['ignore', 'pipe', 'pipe'],\n      windowsVerbatimArguments: true\n    });\n    let buildLastLines = [];\n\n    const logMsg = `Output from Gradle ${this.showGradleLog ? 'will' : 'will not'} be logged`;\n    log.debug(`${logMsg}. To change this, use 'showGradleLog' desired capability`);\n    gradlebuild.on('stream-line', (line) => {\n      if (this.showGradleLog) {\n        if (line.startsWith('[STDERR]')) {\n          gradleLog.warn(line);\n        } else {\n          gradleLog.info(line);\n        }\n      }\n      buildLastLines.push(`${EOL}${line}`);\n      if (buildLastLines.length > GRADLE_MAX_ERROR_LOG_LINES) {\n        buildLastLines = buildLastLines.slice(-GRADLE_MAX_ERROR_LOG_LINES);\n      }\n    });\n\n    try {\n      await gradlebuild.start();\n      await gradlebuild.join();\n    } catch (err) {\n      let msg = `Unable to build Espresso server - ${err.message}\\n` +\n        `Gradle error message:${EOL}${buildLastLines}`;\n      log.errorAndThrow(msg);\n    }\n  }\n}\n\nexport { ServerBuilder, VERSION_KEYS, GRADLE_URL_TEMPLATE, buildServerSigningConfig };\nexport default ServerBuilder;\n"],"file":"lib/server-builder.js","sourceRoot":"../.."}
|
|
Binary file
|
|
@@ -1,18 +1,20 @@
|
|
|
1
1
|
{
|
|
2
|
-
"version":
|
|
2
|
+
"version": 3,
|
|
3
3
|
"artifactType": {
|
|
4
4
|
"type": "APK",
|
|
5
5
|
"kind": "Directory"
|
|
6
6
|
},
|
|
7
7
|
"applicationId": "io.appium.espressoserver.test",
|
|
8
|
-
"variantName": "
|
|
8
|
+
"variantName": "debugAndroidTest",
|
|
9
9
|
"elements": [
|
|
10
10
|
{
|
|
11
11
|
"type": "SINGLE",
|
|
12
12
|
"filters": [],
|
|
13
|
+
"attributes": [],
|
|
13
14
|
"versionCode": 0,
|
|
14
15
|
"versionName": "",
|
|
15
16
|
"outputFile": "app-debug-androidTest.apk"
|
|
16
17
|
}
|
|
17
|
-
]
|
|
18
|
+
],
|
|
19
|
+
"elementType": "File"
|
|
18
20
|
}
|
|
@@ -4,8 +4,8 @@ plugins {
|
|
|
4
4
|
}
|
|
5
5
|
|
|
6
6
|
android {
|
|
7
|
-
|
|
8
|
-
buildToolsVersion
|
|
7
|
+
compileSdk = getIntProperty("appiumCompileSdk", 30)
|
|
8
|
+
buildToolsVersion = getStringProperty("appiumBuildTools", "30.0.3")
|
|
9
9
|
defaultConfig {
|
|
10
10
|
// <instrumentation android:targetPackage=""/>
|
|
11
11
|
applicationId = getStringProperty("appiumTargetPackage", "io.appium.espressoserver")
|
|
@@ -13,8 +13,8 @@ android {
|
|
|
13
13
|
testApplicationId = "io.appium.espressoserver.test"
|
|
14
14
|
testHandleProfiling = false
|
|
15
15
|
testFunctionalTest = false
|
|
16
|
-
|
|
17
|
-
|
|
16
|
+
minSdk = getIntProperty("appiumMinSdk", 21)
|
|
17
|
+
targetSdk = getIntProperty("appiumTargetSdk", 28)
|
|
18
18
|
versionCode = 1
|
|
19
19
|
versionName = "1.0"
|
|
20
20
|
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
|
@@ -22,7 +22,7 @@ android {
|
|
|
22
22
|
|
|
23
23
|
buildTypes {
|
|
24
24
|
getByName("debug") {
|
|
25
|
-
|
|
25
|
+
isZipAlignEnabled = getBooleanProperty("appiumZipAlign", true)
|
|
26
26
|
}
|
|
27
27
|
getByName("release") {
|
|
28
28
|
isMinifyEnabled = false
|
|
@@ -43,19 +43,19 @@ android {
|
|
|
43
43
|
signingConfigs {
|
|
44
44
|
getByName("debug") {
|
|
45
45
|
findProperty("appiumKeystoreFile")?.also {
|
|
46
|
-
storeFile
|
|
46
|
+
storeFile = File(it.toString())
|
|
47
47
|
}
|
|
48
48
|
|
|
49
49
|
findProperty("appiumKeystorePassword")?.also {
|
|
50
|
-
storePassword
|
|
50
|
+
storePassword = it.toString()
|
|
51
51
|
}
|
|
52
52
|
|
|
53
53
|
findProperty("appiumKeyAlias")?.also {
|
|
54
|
-
keyAlias
|
|
54
|
+
keyAlias = it.toString()
|
|
55
55
|
}
|
|
56
56
|
|
|
57
57
|
findProperty("appiumKeyPassword")?.also {
|
|
58
|
-
keyPassword
|
|
58
|
+
keyPassword = it.toString()
|
|
59
59
|
}
|
|
60
60
|
}
|
|
61
61
|
}
|
|
@@ -80,7 +80,7 @@ android {
|
|
|
80
80
|
}
|
|
81
81
|
|
|
82
82
|
packagingOptions {
|
|
83
|
-
|
|
83
|
+
resources.excludes.add("META-INF/**")
|
|
84
84
|
}
|
|
85
85
|
}
|
|
86
86
|
|
|
@@ -140,17 +140,17 @@ tasks.withType<Test> {
|
|
|
140
140
|
|
|
141
141
|
object Version {
|
|
142
142
|
const val kotlin = "1.5.10"
|
|
143
|
-
const val espresso = "3.
|
|
144
|
-
const val testlib = "1.
|
|
145
|
-
const val mocklib = "
|
|
146
|
-
const val gson = "2.8.
|
|
143
|
+
const val espresso = "3.4.0"
|
|
144
|
+
const val testlib = "1.4.0"
|
|
145
|
+
const val mocklib = "2.0.9"
|
|
146
|
+
const val gson = "2.8.9"
|
|
147
147
|
const val uia = "2.2.0"
|
|
148
148
|
const val nanohttpd = "2.3.1"
|
|
149
|
-
const val annotation = "1.
|
|
150
|
-
const val mockito = "
|
|
149
|
+
const val annotation = "1.3.0"
|
|
150
|
+
const val mockito = "4.0.0"
|
|
151
151
|
const val robolectric = "4.5.1"
|
|
152
|
-
const val junit = "4.13"
|
|
153
|
-
const val compose = "1.0.
|
|
152
|
+
const val junit = "4.13.2"
|
|
153
|
+
const val compose = "1.0.5"
|
|
154
154
|
}
|
|
155
155
|
|
|
156
156
|
fun Project.getStringProperty(name: String, default: String): String =
|
|
@@ -19,7 +19,7 @@ import kotlin.test.assertTrue
|
|
|
19
19
|
class `KReflectionUtils Test` {
|
|
20
20
|
|
|
21
21
|
@Test
|
|
22
|
-
fun `
|
|
22
|
+
fun `'invokeMethod' should find methods specific to parameters provided and call it`(){
|
|
23
23
|
val obj = TestClass()
|
|
24
24
|
|
|
25
25
|
var methodResult = ReflectionUtils.invokeInstanceMethod(obj, "plus", null, 1)
|
|
@@ -36,13 +36,13 @@ class `KReflectionUtils Test` {
|
|
|
36
36
|
}
|
|
37
37
|
|
|
38
38
|
@Test
|
|
39
|
-
fun `
|
|
39
|
+
fun `'invokeInstanceMethod' should extract method of an object`() {
|
|
40
40
|
val methodResult = ReflectionUtils.extractMethod(TestClass::class.java, "plus", Int::class.java, Int::class.java)
|
|
41
41
|
assertNotNull(methodResult)
|
|
42
42
|
}
|
|
43
43
|
|
|
44
44
|
@Test
|
|
45
|
-
fun `
|
|
45
|
+
fun `'invokeInstanceMethod' should invoke method on instance of an object`() {
|
|
46
46
|
val obj = TestClass()
|
|
47
47
|
val methodResult = ReflectionUtils.invokeInstanceMethod(obj, "plus", 2, 2)
|
|
48
48
|
assertTrue(methodResult is Number)
|
|
@@ -56,7 +56,7 @@ class `KReflectionUtils Test` {
|
|
|
56
56
|
}
|
|
57
57
|
|
|
58
58
|
@Test
|
|
59
|
-
fun `should parse Driver Atoms
|
|
59
|
+
fun `should parse Driver Atoms 'findElement'`() {
|
|
60
60
|
val findElementAtom = invokeStaticMethod(DriverAtoms::class.java,
|
|
61
61
|
"findElement", Locator.ID, "some Identifier")
|
|
62
62
|
assertTrue(findElementAtom is Atom<*>)
|
package/espresso-server/app/src/test/java/io/appium/espressoserver/test/model/MobileClickTest.kt
CHANGED
|
@@ -15,7 +15,7 @@ import kotlin.test.assertEquals
|
|
|
15
15
|
class `mobile clickAction test` {
|
|
16
16
|
|
|
17
17
|
@Test
|
|
18
|
-
fun `should parse
|
|
18
|
+
fun `should parse 'MobileClickParams' and set defaults if some params not provided` () {
|
|
19
19
|
val jsonElement = JsonObject()
|
|
20
20
|
val clickActionParams = MobileClickActionParams.MobileClickActionParamsDeserializer()
|
|
21
21
|
.deserialize(jsonElement, null, null)
|
|
@@ -27,7 +27,7 @@ class `mobile clickAction test` {
|
|
|
27
27
|
}
|
|
28
28
|
|
|
29
29
|
@Test
|
|
30
|
-
fun `should parse
|
|
30
|
+
fun `should parse 'MobileClickParams' and set values if all params provided` () {
|
|
31
31
|
val jsonElement = JsonObject()
|
|
32
32
|
jsonElement.add("inputDevice", JsonPrimitive(2))
|
|
33
33
|
jsonElement.add("buttonState", JsonPrimitive("3"))
|
package/espresso-server/app/src/test/java/io/appium/espressoserver/test/model/MobileSwipeTest.kt
CHANGED
|
@@ -15,7 +15,7 @@ import kotlin.test.assertEquals
|
|
|
15
15
|
class `mobile swipe test` {
|
|
16
16
|
|
|
17
17
|
@Test
|
|
18
|
-
fun `should parse
|
|
18
|
+
fun `should parse 'MobileSwipeActionParams' with 'direction' property` () {
|
|
19
19
|
val jsonElement = JsonObject()
|
|
20
20
|
jsonElement.add("direction", JsonPrimitive("DOWN"))
|
|
21
21
|
val mobileSwipeActionParams = MobileSwipeActionParamsDeserializer()
|
|
@@ -24,7 +24,7 @@ class `mobile swipe test` {
|
|
|
24
24
|
}
|
|
25
25
|
|
|
26
26
|
@Test
|
|
27
|
-
fun `should parse
|
|
27
|
+
fun `should parse 'MobileSwipeActionParams' with 'swiper' property and use default parameters` () {
|
|
28
28
|
val jsonElement = JsonObject()
|
|
29
29
|
jsonElement.add("swiper", JsonPrimitive("FAST"))
|
|
30
30
|
val mobileSwipeActionParams = MobileSwipeActionParamsDeserializer()
|
|
@@ -36,7 +36,7 @@ class `mobile swipe test` {
|
|
|
36
36
|
}
|
|
37
37
|
|
|
38
38
|
@Test
|
|
39
|
-
fun `should parse
|
|
39
|
+
fun `should parse 'MobileSwipeActionParams' with 'swiper' property plus other parameters` () {
|
|
40
40
|
val jsonElement = JsonObject()
|
|
41
41
|
jsonElement.add("swiper", JsonPrimitive("FAST"))
|
|
42
42
|
jsonElement.add("startCoordinates", JsonPrimitive("TOP_LEFT"))
|
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
// Top-level build file where you can add configuration options common to all sub-projects/modules.
|
|
2
2
|
|
|
3
3
|
buildscript {
|
|
4
|
+
|
|
4
5
|
val appiumKotlin =
|
|
5
6
|
properties.getOrDefault("appiumKotlin", "1.5.10")
|
|
6
7
|
|
|
7
8
|
val appiumAndroidGradlePlugin =
|
|
8
|
-
properties.getOrDefault("appiumAndroidGradlePlugin", "
|
|
9
|
+
properties.getOrDefault("appiumAndroidGradlePlugin", "7.0.3")
|
|
9
10
|
|
|
10
11
|
repositories {
|
|
11
12
|
google()
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
distributionBase=GRADLE_USER_HOME
|
|
2
2
|
distributionPath=wrapper/dists
|
|
3
|
-
distributionUrl=https\://services.gradle.org/distributions/gradle-
|
|
3
|
+
distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-all.zip
|
|
4
4
|
zipStoreBase=GRADLE_USER_HOME
|
|
5
5
|
zipStorePath=wrapper/dists
|
package/lib/driver.js
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import _ from 'lodash';
|
|
2
|
+
import path from 'path';
|
|
2
3
|
import { BaseDriver, errors, isErrorType, DeviceSettings} from '@appium/base-driver';
|
|
3
4
|
import { EspressoRunner, TEST_APK_PKG } from './espresso-runner';
|
|
4
|
-
import { fs } from '@appium/support';
|
|
5
|
+
import { fs, tempDir, zip } from '@appium/support';
|
|
5
6
|
import logger from './logger';
|
|
6
7
|
import commands from './commands';
|
|
7
8
|
import { DEFAULT_ADB_PORT } from 'appium-adb';
|
|
@@ -119,8 +120,10 @@ const CHROME_NO_PROXY = [
|
|
|
119
120
|
['POST', new RegExp('^/session/[^/]+/se/log')],
|
|
120
121
|
];
|
|
121
122
|
|
|
122
|
-
const APP_EXTENSION = '.apk';
|
|
123
123
|
|
|
124
|
+
const APK_EXT = '.apk';
|
|
125
|
+
const AAB_EXT = '.aab';
|
|
126
|
+
const SUPPORTED_EXTENSIONS = [APK_EXT, AAB_EXT];
|
|
124
127
|
|
|
125
128
|
class EspressoDriver extends BaseDriver {
|
|
126
129
|
constructor (opts = {}, shouldValidateCaps = true) {
|
|
@@ -177,12 +180,12 @@ class EspressoDriver extends BaseDriver {
|
|
|
177
180
|
|
|
178
181
|
if (this.isChromeSession) {
|
|
179
182
|
if (this.opts.app) {
|
|
180
|
-
logger.warn(`
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
`);
|
|
183
|
+
logger.warn(`'browserName' capability will be ignored`);
|
|
184
|
+
logger.warn(`Chrome browser cannot be run in Espresso sessions because Espresso automation doesn't ` +
|
|
185
|
+
`have permission to access Chrome`);
|
|
184
186
|
} else {
|
|
185
|
-
logger.errorAndThrow(`Chrome browser sessions cannot be run in Espresso because Espresso
|
|
187
|
+
logger.errorAndThrow(`Chrome browser sessions cannot be run in Espresso because Espresso ` +
|
|
188
|
+
`automation doesn't have permission to access Chrome`);
|
|
186
189
|
}
|
|
187
190
|
}
|
|
188
191
|
|
|
@@ -191,25 +194,41 @@ class EspressoDriver extends BaseDriver {
|
|
|
191
194
|
this.addWipeDataToAvdArgs();
|
|
192
195
|
}
|
|
193
196
|
|
|
197
|
+
this.opts.systemPort = this.opts.systemPort
|
|
198
|
+
|| await findAPortNotInUse(SYSTEM_PORT_RANGE[0], SYSTEM_PORT_RANGE[1]);
|
|
199
|
+
this.opts.adbPort = this.opts.adbPort || DEFAULT_ADB_PORT;
|
|
200
|
+
// get device udid for this session
|
|
201
|
+
const {udid, emPort} = await helpers.getDeviceInfoFromCaps(this.opts);
|
|
202
|
+
this.opts.udid = udid;
|
|
203
|
+
this.opts.emPort = emPort;
|
|
204
|
+
// now that we know our java version and device info, we can create our
|
|
205
|
+
// ADB instance
|
|
206
|
+
this.adb = await androidHelpers.createADB(this.opts);
|
|
207
|
+
|
|
194
208
|
if (this.opts.app) {
|
|
195
209
|
// find and copy, or download and unzip an app url or path
|
|
196
|
-
this.opts.app = await this.helpers.configureApp(this.opts.app,
|
|
197
|
-
|
|
210
|
+
this.opts.app = await this.helpers.configureApp(this.opts.app, {
|
|
211
|
+
onPostProcess: this.onPostConfigureApp.bind(this),
|
|
212
|
+
supportedExtensions: SUPPORTED_EXTENSIONS
|
|
213
|
+
});
|
|
198
214
|
} else if (this.appOnDevice) {
|
|
199
215
|
// the app isn't an actual app file but rather something we want to
|
|
200
216
|
// assume is on the device and just launch via the appPackage
|
|
201
217
|
logger.info(`App file was not listed, instead we're going to run ` +
|
|
202
218
|
`${this.opts.appPackage} directly on the device`);
|
|
203
|
-
await this.
|
|
219
|
+
if (!await this.adb.isAppInstalled(this.opts.appPackage)) {
|
|
220
|
+
logger.errorAndThrow(`Could not find the package '${this.opts.appPackage}' ` +
|
|
221
|
+
`installed on the device`);
|
|
222
|
+
}
|
|
204
223
|
}
|
|
205
224
|
|
|
206
|
-
this.opts.systemPort = this.opts.systemPort || await findAPortNotInUse(SYSTEM_PORT_RANGE[0], SYSTEM_PORT_RANGE[1]);
|
|
207
|
-
this.opts.adbPort = this.opts.adbPort || DEFAULT_ADB_PORT;
|
|
208
225
|
await this.startEspressoSession();
|
|
209
226
|
return [sessionId, caps];
|
|
210
227
|
} catch (e) {
|
|
211
228
|
await this.deleteSession();
|
|
212
|
-
e.message += '.
|
|
229
|
+
e.message += `${_.endsWith(e.message, '.') ? '' : '.'} Check ` +
|
|
230
|
+
'https://github.com/appium/appium-espresso-driver#troubleshooting ' +
|
|
231
|
+
'regarding advanced session startup troubleshooting.';
|
|
213
232
|
if (isErrorType(e, errors.SessionNotCreatedError)) {
|
|
214
233
|
throw e;
|
|
215
234
|
}
|
|
@@ -219,6 +238,102 @@ class EspressoDriver extends BaseDriver {
|
|
|
219
238
|
}
|
|
220
239
|
}
|
|
221
240
|
|
|
241
|
+
/**
|
|
242
|
+
* Unzip the given app path and return the first package that has SUPPORTED_EXTENSIONS
|
|
243
|
+
* in the archived file.
|
|
244
|
+
*
|
|
245
|
+
* @param {string} appPath The path to app file.
|
|
246
|
+
* @returns {string} Retuns the path to an unzipped app file path.
|
|
247
|
+
* @throws Raise an exception if the zip did not have any SUPPORTED_EXTENSIONS packages.
|
|
248
|
+
*/
|
|
249
|
+
async unzipApp (appPath) {
|
|
250
|
+
const useSystemUnzipEnv = process.env.APPIUM_PREFER_SYSTEM_UNZIP;
|
|
251
|
+
const useSystemUnzip = _.isEmpty(useSystemUnzipEnv)
|
|
252
|
+
|| !['0', 'false'].includes(_.toLower(useSystemUnzipEnv));
|
|
253
|
+
const tmpRoot = await tempDir.openDir();
|
|
254
|
+
await zip.extractAllTo(appPath, tmpRoot, {useSystemUnzip});
|
|
255
|
+
|
|
256
|
+
const globPattern = `**/*.+(${SUPPORTED_EXTENSIONS.map((ext) => ext.replace(/^\./, '')).join('|')})`;
|
|
257
|
+
const sortedBundleItems = (await fs.glob(globPattern, {
|
|
258
|
+
cwd: tmpRoot,
|
|
259
|
+
strict: false,
|
|
260
|
+
})).sort((a, b) => a.split(path.sep).length - b.split(path.sep).length);
|
|
261
|
+
if (sortedBundleItems.length === 0) {
|
|
262
|
+
// no expected packages in the zip
|
|
263
|
+
logger.errorAndThrow(`${this.opts.app} did not have any of '${SUPPORTED_EXTENSIONS.join(', ')}' ` +
|
|
264
|
+
`extension packages. Please make sure the provided .zip archive contains at least one valid application package.`);
|
|
265
|
+
}
|
|
266
|
+
const unzippedAppPath = path.join(tmpRoot, _.first(sortedBundleItems));
|
|
267
|
+
logger.debug(`'${unzippedAppPath}' is the unzipped file from '${appPath}'`);
|
|
268
|
+
return unzippedAppPath;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
async onPostConfigureApp ({cachedAppInfo, isUrl, appPath}) {
|
|
272
|
+
const presignApp = async (appLocation) => {
|
|
273
|
+
if (this.opts.noSign) {
|
|
274
|
+
logger.info('Skipping application signing because noSign capability is set to true. ' +
|
|
275
|
+
'Having the application under test with improper signature/non-signed will cause ' +
|
|
276
|
+
'Espresso automation startup failure.');
|
|
277
|
+
} else if (!await this.adb.checkApkCert(appLocation, this.opts.appPackage)) {
|
|
278
|
+
await this.adb.sign(appLocation, this.opts.appPackage);
|
|
279
|
+
}
|
|
280
|
+
};
|
|
281
|
+
|
|
282
|
+
const hasApkExt = (appPath) => _.endsWith(_.toLower(appPath), APK_EXT);
|
|
283
|
+
const hasAabExt = (appPath) => _.endsWith(_.toLower(appPath), AAB_EXT);
|
|
284
|
+
const extractUniversalApk = async (shouldExtract, appPath) =>
|
|
285
|
+
shouldExtract ? appPath : await this.adb.extractUniversalApk(appPath);
|
|
286
|
+
|
|
287
|
+
let pathInCache = null;
|
|
288
|
+
let isResultAppPathAlreadyCached = false;
|
|
289
|
+
if (_.isPlainObject(cachedAppInfo)) {
|
|
290
|
+
const packageHash = await fs.hash(appPath);
|
|
291
|
+
if (packageHash === cachedAppInfo.packageHash && await fs.exists(cachedAppInfo.fullPath)) {
|
|
292
|
+
logger.info(`Using '${cachedAppInfo.fullPath}' which is cached from '${appPath}'`);
|
|
293
|
+
isResultAppPathAlreadyCached = true;
|
|
294
|
+
pathInCache = cachedAppInfo.fullPath;
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
// appPath can be .zip, .apk or .aab
|
|
299
|
+
const isApk = hasApkExt(appPath);
|
|
300
|
+
// Only local .apk files that are available in-place should not be cached
|
|
301
|
+
const shouldResultAppPathBeCached = !isApk || (isApk && isUrl);
|
|
302
|
+
|
|
303
|
+
if (!isResultAppPathAlreadyCached) {
|
|
304
|
+
if (shouldResultAppPathBeCached) {
|
|
305
|
+
// .zip, .aab or downloaded .apk
|
|
306
|
+
|
|
307
|
+
let unzippedAppPath;
|
|
308
|
+
let isUnzippedApk = false;
|
|
309
|
+
if (!(hasApkExt(appPath) || hasAabExt(appPath))) {
|
|
310
|
+
unzippedAppPath = await this.unzipApp(appPath);
|
|
311
|
+
isUnzippedApk = hasApkExt(unzippedAppPath);
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
// unzippedAppPath or appPath has SUPPORTED_EXTENSIONS.
|
|
315
|
+
pathInCache = unzippedAppPath
|
|
316
|
+
? await extractUniversalApk(isUnzippedApk, unzippedAppPath)
|
|
317
|
+
: await extractUniversalApk(isApk, appPath);
|
|
318
|
+
|
|
319
|
+
if (!isApk && isUrl) {
|
|
320
|
+
// Clean up the temporarily downloaded .aab or .zip package
|
|
321
|
+
await fs.rimraf(appPath);
|
|
322
|
+
}
|
|
323
|
+
if (hasAabExt(unzippedAppPath)) {
|
|
324
|
+
// Cleanup the local unzipped .aab file
|
|
325
|
+
await fs.rimraf(unzippedAppPath);
|
|
326
|
+
}
|
|
327
|
+
await presignApp(pathInCache);
|
|
328
|
+
} else if (isApk) {
|
|
329
|
+
// It is probably not the best idea to modify the provided app in-place,
|
|
330
|
+
// but this is how it was always working
|
|
331
|
+
await presignApp(appPath);
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
return shouldResultAppPathBeCached ? {appPath: pathInCache} : false;
|
|
335
|
+
}
|
|
336
|
+
|
|
222
337
|
get driverData () {
|
|
223
338
|
// TODO fille out resource info here
|
|
224
339
|
return {};
|
|
@@ -257,15 +372,6 @@ class EspressoDriver extends BaseDriver {
|
|
|
257
372
|
async startEspressoSession () {
|
|
258
373
|
logger.info(`EspressoDriver version: ${version}`);
|
|
259
374
|
|
|
260
|
-
// get device udid for this session
|
|
261
|
-
let {udid, emPort} = await helpers.getDeviceInfoFromCaps(this.opts);
|
|
262
|
-
this.opts.udid = udid;
|
|
263
|
-
this.opts.emPort = emPort;
|
|
264
|
-
|
|
265
|
-
// now that we know our java version and device info, we can create our
|
|
266
|
-
// ADB instance
|
|
267
|
-
this.adb = await androidHelpers.createADB(this.opts);
|
|
268
|
-
|
|
269
375
|
// Read https://github.com/appium/appium-android-driver/pull/461 what happens if ther is no setHiddenApiPolicy for Android P+
|
|
270
376
|
if (await this.adb.getApiLevel() >= 28) { // Android P
|
|
271
377
|
logger.warn('Relaxing hidden api policy');
|
|
@@ -438,13 +544,6 @@ class EspressoDriver extends BaseDriver {
|
|
|
438
544
|
await this.adb.uninstallApk(this.opts.appPackage);
|
|
439
545
|
}
|
|
440
546
|
if (this.opts.app) {
|
|
441
|
-
if (this.opts.noSign) {
|
|
442
|
-
logger.info('Skipping application signing because noSign capability is set to true. ' +
|
|
443
|
-
'Having the application under test with improper signature/non-signed will cause ' +
|
|
444
|
-
'Espresso automation startup failure.');
|
|
445
|
-
} else if (!await this.adb.checkApkCert(this.opts.app, this.opts.appPackage)) {
|
|
446
|
-
await this.adb.sign(this.opts.app, this.opts.appPackage);
|
|
447
|
-
}
|
|
448
547
|
await helpers.installApk(this.adb, this.opts);
|
|
449
548
|
}
|
|
450
549
|
if (this.opts.skipServerInstallation) {
|
|
@@ -525,14 +624,6 @@ class EspressoDriver extends BaseDriver {
|
|
|
525
624
|
}
|
|
526
625
|
}
|
|
527
626
|
|
|
528
|
-
// TODO method is duplicated from uiautomator2
|
|
529
|
-
async checkAppPresent () {
|
|
530
|
-
logger.debug('Checking whether app is actually present');
|
|
531
|
-
if (!(await fs.exists(this.opts.app))) {
|
|
532
|
-
logger.errorAndThrow(`Could not find app apk at '${this.opts.app}'`);
|
|
533
|
-
}
|
|
534
|
-
}
|
|
535
|
-
|
|
536
627
|
async onSettingsUpdate () {
|
|
537
628
|
// intentionally do nothing here, since commands.updateSettings proxies
|
|
538
629
|
// settings to the espresso server already
|
|
@@ -567,6 +658,10 @@ class EspressoDriver extends BaseDriver {
|
|
|
567
658
|
get isChromeSession () {
|
|
568
659
|
return helpers.isChromeBrowser(this.opts.browserName);
|
|
569
660
|
}
|
|
661
|
+
|
|
662
|
+
get appOnDevice () {
|
|
663
|
+
return !this.opts.app && this.helpers.isPackageOrBundle(this.opts.appPackage);
|
|
664
|
+
}
|
|
570
665
|
}
|
|
571
666
|
|
|
572
667
|
// first add the android-driver commands which we will fall back to
|
package/lib/server-builder.js
CHANGED
|
@@ -22,7 +22,7 @@ const VERSION_KEYS = [
|
|
|
22
22
|
'kotlin',
|
|
23
23
|
'sourceCompatibility',
|
|
24
24
|
'targetCompatibility',
|
|
25
|
-
'jvmTarget'
|
|
25
|
+
'jvmTarget'
|
|
26
26
|
];
|
|
27
27
|
|
|
28
28
|
const gradleLog = logger.getLogger('Gradle');
|
|
@@ -163,6 +163,7 @@ class ServerBuilder {
|
|
|
163
163
|
const gradlebuild = new SubProcess(cmd, args, {
|
|
164
164
|
cwd: this.serverPath,
|
|
165
165
|
stdio: ['ignore', 'pipe', 'pipe'],
|
|
166
|
+
windowsVerbatimArguments: true
|
|
166
167
|
});
|
|
167
168
|
let buildLastLines = [];
|
|
168
169
|
|