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.
@@ -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":";;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA,MAAMA,kBAAkB,GAAG,QAA3B;AACA,MAAMC,iBAAiB,GAAG,kBAA1B;AACA,MAAMC,mBAAmB,GAAG,oEAA5B;;AACA,MAAMC,0BAA0B,GAAG,EAAnC;AACA,MAAMC,qBAAqB,GAAG,CAAC,2BAAD,EAA8B,mCAA9B,CAA9B;AAEA,MAAMC,YAAY,GAAG,CACnBL,kBADmB,EAEnB,qBAFmB,EAGnB,YAHmB,EAInB,YAJmB,EAKnB,QALmB,EAMnB,WANmB,EAOnB,QAPmB,EAQnB,qBARmB,EASnB,qBATmB,EAUnB,WAVmB,CAArB;;;AAaA,MAAMM,SAAS,GAAGC,gBAAOC,SAAP,CAAiB,QAAjB,CAAlB;;AAEA,SAASC,wBAAT,CAAmCC,IAAnC,EAAyC;AACvC,SAAO;AACLC,IAAAA,QAAQ,EAAE,IADL;AAELC,IAAAA,YAAY,EAAEF,IAAI,CAACE,YAFd;AAGLC,IAAAA,gBAAgB,EAAEH,IAAI,CAACG,gBAHlB;AAILC,IAAAA,QAAQ,EAAEJ,IAAI,CAACI,QAJV;AAKLC,IAAAA,WAAW,EAAEL,IAAI,CAACK;AALb,GAAP;AAOD;;AAED,MAAMC,aAAN,CAAoB;AAClBC,EAAAA,WAAW,CAAEP,IAAI,GAAG,EAAT,EAAa;AACtB,SAAKQ,UAAL,GAAkBR,IAAI,CAACQ,UAAvB;AACA,SAAKC,aAAL,GAAqBT,IAAI,CAACS,aAA1B;AAEA,UAAMC,kBAAkB,GAAGV,IAAI,CAACU,kBAAL,IAA2B,EAAtD;AAEA,UAAMC,oBAAoB,GAAGD,kBAAkB,CAACE,aAAnB,IAAoC,EAAjE;AACA,SAAKC,cAAL,GAAsBC,gBAAEC,MAAF,CAASJ,oBAAT,EAA+B,CAACK,GAAD,EAAMC,KAAN,EAAaC,GAAb,KAAqB;AACxE,UAAIvB,YAAY,CAACwB,QAAb,CAAsBD,GAAtB,CAAJ,EAAgC;AAC9BF,QAAAA,GAAG,CAACE,GAAD,CAAH,GAAWD,KAAX;AACD,OAFD,MAEO;AACLG,wBAAIC,IAAJ,CAAU,mBAAkBH,GAAI,oDAAhC;AACD;;AACD,aAAOF,GAAP;AACD,KAPqB,EAOnB,EAPmB,CAAtB;AASA,SAAKM,cAAL,GAAsBtB,IAAI,CAACsB,cAA3B;AACA,SAAKC,aAAL,GAAqBvB,IAAI,CAACuB,aAA1B;;AAEA,SAAK,MAAMC,QAAX,IAAuB9B,qBAAvB,EAA8C;AAC5C,WAAK8B,QAAL,IAAiBd,kBAAkB,CAACc,QAAD,CAAlB,IAAgC,EAAjD;AACD;AACF;;AAEU,QAALC,KAAK,GAAI;AACb,QAAI,KAAKZ,cAAL,CAAoBvB,kBAApB,CAAJ,EAA6C;AAC3C,YAAM,KAAKoC,uBAAL,CAA6B,KAAKb,cAAL,CAAoBvB,kBAApB,CAA7B,CAAN;AACD;;AAED,UAAM,KAAKqC,4BAAL,EAAN;AAEA,UAAM,KAAKC,eAAL,EAAN;AACD;;AAEDC,EAAAA,UAAU,GAAI;AACZ,UAAMC,GAAG,GAAGC,gBAAOC,SAAP,KAAqB,aAArB,GAAqC,WAAjD;;AACA,UAAMC,aAAa,GAAG,CAACf,GAAD,EAAMD,KAAN,KAAgBA,KAAK,GAAI,KAAIC,GAAI,IAAGD,KAAM,EAArB,GAAyB,IAApE;;AACA,QAAIjB,IAAI,GAAGL,YAAY,CACpBuC,MADQ,CACAhB,GAAD,IAASA,GAAG,KAAK5B,kBADhB,EAER6C,GAFQ,CAEHjB,GAAD,IAAS;AACZ,YAAMkB,aAAa,GAAG,KAAKvB,cAAL,CAAoBK,GAApB,CAAtB;AACA,YAAMmB,cAAc,GAAI,SAAQnB,GAAG,CAACoB,MAAJ,CAAW,CAAX,EAAcC,WAAd,EAA4B,GAAErB,GAAG,CAACsB,KAAJ,CAAU,CAAV,CAAa,EAA3E;AACA,aAAOP,aAAa,CAACI,cAAD,EAAiBD,aAAjB,CAApB;AACD,KANQ,EAORF,MAPQ,CAODO,OAPC,CAAX;;AASA,QAAI,KAAKlB,aAAT,EAAwB;AACtBvB,MAAAA,IAAI,CAAC0C,IAAL,CAAU,GACR5B,gBAAE6B,IAAF,CAAO,KAAKpB,aAAZ,EACCY,GADD,CACMjB,GAAD,IAAS,CAAE,SAAQJ,gBAAE8B,UAAF,CAAa1B,GAAb,CAAkB,EAA5B,EAA+B,KAAKK,aAAL,CAAmBL,GAAnB,CAA/B,CADd,EAECiB,GAFD,CAEK,CAAC,CAACU,CAAD,EAAIC,CAAJ,CAAD,KAAYb,aAAa,CAACY,CAAD,EAAIC,CAAJ,CAF9B,EAGCZ,MAHD,CAGQO,OAHR,CADF;AAMD;;AAED,QAAI,KAAKnB,cAAT,EAAyB;AACvBtB,MAAAA,IAAI,CAAC0C,IAAL,CAAUT,aAAa,CAAC,qBAAD,EAAwB,KAAKX,cAA7B,CAAvB;AACD;;AACDtB,IAAAA,IAAI,CAAC0C,IAAL,CAAU,yBAAV;AAEA,WAAO;AAACZ,MAAAA,GAAD;AAAM9B,MAAAA;AAAN,KAAP;AACD;;AAE4B,QAAvB0B,uBAAuB,CAAEqB,OAAF,EAAW;AACtC,UAAMC,cAAc,GAAGC,cAAKC,OAAL,CAAa,KAAK1C,UAAlB,EAA8B,QAA9B,EAAwC,SAAxC,EAAmD,2BAAnD,CAAvB;;AACA,UAAM2C,kBAAkB,GAAG,MAAMC,YAAGC,QAAH,CAAYL,cAAZ,EAA4B,MAA5B,CAAjC;AACA,UAAMM,aAAa,GAAG,KAAKC,mBAAL,CAAyBJ,kBAAzB,EAA6CJ,OAA7C,CAAtB;AACA,UAAMK,YAAGI,SAAH,CAAaR,cAAb,EAA6BM,aAA7B,EAA4C,MAA5C,CAAN;AACD;;AAEDC,EAAAA,mBAAmB,CAAEE,iBAAF,EAAqBV,OAArB,EAA8B;AAC/C,WAAOU,iBAAiB,CAACC,OAAlB,CACL,IAAIC,MAAJ,CAAY,KAAI7C,gBAAE8C,YAAF,CAAerE,iBAAf,CAAkC,MAAlD,EAAyD,IAAzD,CADK,EAEJ,KAAIC,mBAAmB,CAACkE,OAApB,CAA4B,SAA5B,EAAuCX,OAAvC,CAAgD,EAFhD,CAAP;AAID;;AAEiC,QAA5BpB,4BAA4B,GAAI;AACpC,QAAIkC,iBAAiB,GAAG,KAAxB;;AACA,SAAK,MAAMrC,QAAX,IAAuB9B,qBAAvB,EAA8C;AAC5C,UAAI,CAACoB,gBAAEgD,OAAF,CAAU,KAAKtC,QAAL,CAAV,CAAL,EAAgC;AAC9B,cAAM,IAAIuC,KAAJ,CAAW,IAAGvC,QAAS,oBAAvB,CAAN;AACD;;AACD,UAAIV,gBAAEkD,OAAF,CAAU,KAAKxC,QAAL,EAAeU,MAAf,CAAuB+B,IAAD,IAAUnD,gBAAEoD,IAAF,CAAOD,IAAP,CAAhC,CAAV,CAAJ,EAA8D;AAC5D;AACD;;AAED,WAAK,MAAME,GAAX,IAAkB,KAAK3C,QAAL,CAAlB,EAAkC;AAChC,YAAI,WAAW4C,IAAX,CAAgBD,GAAhB,CAAJ,EAA0B;AACxB,gBAAM,IAAIJ,KAAJ,CAAU,+DACb,+CAA8CI,GAAI,EAD/C,CAAN;AAED;AACF;;AACDN,MAAAA,iBAAiB,GAAG,IAApB;AACD;;AACD,QAAI,CAACA,iBAAL,EAAwB;AACtB;AACD;;AAED,UAAMQ,SAAS,GAAGpB,cAAKC,OAAL,CAAa,KAAK1C,UAAlB,EAA8B,KAA9B,EAAqC,kBAArC,CAAlB;;AACA,QAAI8D,aAAa,GAAG,MAAMlB,YAAGC,QAAH,CAAYgB,SAAZ,EAAuB,MAAvB,CAA1B;;AACA,SAAK,MAAM7C,QAAX,IAAuB9B,qBAAvB,EAA8C;AAC5C,YAAM6E,MAAM,GAAG/C,QAAQ,KAAK9B,qBAAqB,CAAC,CAAD,CAAlC,GACX,gBADW,GAEX,2BAFJ;AAGA,YAAM8E,IAAI,GAAG,KAAKhD,QAAL,EACVU,MADU,CACF+B,IAAD,IAAUnD,gBAAEoD,IAAF,CAAOD,IAAP,CADP,EAEV9B,GAFU,CAEL8B,IAAD,IAAW,GAAEM,MAAO,KAAIN,IAAK,IAFvB,CAAb;;AAGA,UAAInD,gBAAEkD,OAAF,CAAUQ,IAAV,CAAJ,EAAqB;AACnB;AACD;;AAEDpD,sBAAIqD,IAAJ,CAAU,wBAAuBjD,QAAS,yBAAwBgD,IAAK,EAAvE;;AACAF,MAAAA,aAAa,GAAG,kCAAsBA,aAAtB,EAAqC9C,QAArC,EAA+CgD,IAA/C,CAAhB;AACD;;AACD,UAAMpB,YAAGI,SAAH,CAAaa,SAAb,EAAwBC,aAAxB,EAAuC,MAAvC,CAAN;AACD;;AAEoB,QAAf1C,eAAe,GAAI;AACvB,UAAM;AAACE,MAAAA,GAAD;AAAM9B,MAAAA;AAAN,QAAc,KAAK6B,UAAL,EAApB;;AACAT,oBAAIsD,KAAJ,CAAW,iCAAgC5C,GAAI,IAAG9B,IAAI,CAAC2E,IAAL,CAAU,GAAV,CAAe,IAAvD,GACP,iBAAgB,KAAKnE,UAAW,GADnC;;AAEA,UAAMoE,WAAW,GAAG,IAAIC,wBAAJ,CAAe/C,GAAf,EAAoB9B,IAApB,EAA0B;AAC5C8E,MAAAA,GAAG,EAAE,KAAKtE,UADkC;AAE5CuE,MAAAA,KAAK,EAAE,CAAC,QAAD,EAAW,MAAX,EAAmB,MAAnB;AAFqC,KAA1B,CAApB;AAIA,QAAIC,cAAc,GAAG,EAArB;AAEA,UAAMC,MAAM,GAAI,sBAAqB,KAAKxE,aAAL,GAAqB,MAArB,GAA8B,UAAW,YAA9E;;AACAW,oBAAIsD,KAAJ,CAAW,GAAEO,MAAO,0DAApB;;AACAL,IAAAA,WAAW,CAACM,EAAZ,CAAe,aAAf,EAA+BjB,IAAD,IAAU;AACtC,UAAI,KAAKxD,aAAT,EAAwB;AACtB,YAAIwD,IAAI,CAACkB,UAAL,CAAgB,UAAhB,CAAJ,EAAiC;AAC/BvF,UAAAA,SAAS,CAACyB,IAAV,CAAe4C,IAAf;AACD,SAFD,MAEO;AACLrE,UAAAA,SAAS,CAAC6E,IAAV,CAAeR,IAAf;AACD;AACF;;AACDe,MAAAA,cAAc,CAACtC,IAAf,CAAqB,GAAE0C,OAAI,GAAEnB,IAAK,EAAlC;;AACA,UAAIe,cAAc,CAACK,MAAf,GAAwB5F,0BAA5B,EAAwD;AACtDuF,QAAAA,cAAc,GAAGA,cAAc,CAACxC,KAAf,CAAqB,CAAC/C,0BAAtB,CAAjB;AACD;AACF,KAZD;;AAcA,QAAI;AACF,YAAMmF,WAAW,CAACU,KAAZ,EAAN;AACA,YAAMV,WAAW,CAACD,IAAZ,EAAN;AACD,KAHD,CAGE,OAAOY,GAAP,EAAY;AACZ,UAAIC,GAAG,GAAI,qCAAoCD,GAAG,CAACE,OAAQ,IAAjD,GACP,wBAAuBL,OAAI,GAAEJ,cAAe,EAD/C;;AAEA5D,sBAAIsE,aAAJ,CAAkBF,GAAlB;AACD;AACF;;AAzJiB;;;eA6JLlF,a","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":"../.."}
@@ -1,18 +1,20 @@
1
1
  {
2
- "version": 2,
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": "processDebugAndroidTestResources",
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
- compileSdkVersion(getIntProperty("appiumCompileSdk", 30))
8
- buildToolsVersion(getStringProperty("appiumBuildTools", "30.0.3"))
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
- minSdkVersion(getIntProperty("appiumMinSdk", 21))
17
- targetSdkVersion(getIntProperty("appiumTargetSdk", 28))
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
- zipAlignEnabled(getBooleanProperty("appiumZipAlign", true))
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(File(it.toString()))
46
+ storeFile = File(it.toString())
47
47
  }
48
48
 
49
49
  findProperty("appiumKeystorePassword")?.also {
50
- storePassword(it.toString())
50
+ storePassword = it.toString()
51
51
  }
52
52
 
53
53
  findProperty("appiumKeyAlias")?.also {
54
- keyAlias(it.toString())
54
+ keyAlias = it.toString()
55
55
  }
56
56
 
57
57
  findProperty("appiumKeyPassword")?.also {
58
- keyPassword(it.toString())
58
+ keyPassword = it.toString()
59
59
  }
60
60
  }
61
61
  }
@@ -80,7 +80,7 @@ android {
80
80
  }
81
81
 
82
82
  packagingOptions {
83
- exclude("META-INF/**")
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.3.0"
144
- const val testlib = "1.3.0"
145
- const val mocklib = "1.7.4"
146
- const val gson = "2.8.6"
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.1.0"
150
- const val mockito = "2.8.9"
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.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 `"invokeMethod" should find methods specific to parameters provided and call it`(){
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 `"invokeInstanceMethod" should extract method of an object`() {
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 `"invokeInstanceMethod" should invoke method on instance of an object`() {
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 "findElement"`() {
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<*>)
@@ -15,7 +15,7 @@ import kotlin.test.assertEquals
15
15
  class `mobile clickAction test` {
16
16
 
17
17
  @Test
18
- fun `should parse "MobileClickParams" and set defaults if some params not provided` () {
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 "MobileClickParams" and set values if all params provided` () {
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"))
@@ -15,7 +15,7 @@ import kotlin.test.assertEquals
15
15
  class `mobile swipe test` {
16
16
 
17
17
  @Test
18
- fun `should parse "MobileSwipeActionParams" with 'direction' property` () {
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 "MobileSwipeActionParams" with 'swiper' property and use default parameters` () {
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 "MobileSwipeActionParams" with 'swiper' property plus other parameters` () {
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", "4.1.1")
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-6.8.3-all.zip
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
- 'browserName' capability will be ignored.
182
- Chrome browser cannot be run in Espresso sessions because Espresso automation doesn't have permission to access Chrome.
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 automation doesn't have permission to access Chrome`);
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, APP_EXTENSION);
197
- await this.checkAppPresent();
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.checkPackagePresent();
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 += '. Check https://github.com/appium/appium-espresso-driver#troubleshooting regarding advanced session startup troubleshooting.';
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
@@ -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