UNPKG

@appium/support

Version:

Support libs used across Appium packages

150 lines (137 loc) 4.47 kB
// @ts-check import _ from 'lodash'; import {homedir} from 'os'; import path from 'path'; import readPkg from 'read-pkg'; import * as semver from 'semver'; /** * Path to the default `APPIUM_HOME` dir (`~/.appium`). * @type {string} */ export const DEFAULT_APPIUM_HOME = path.resolve(homedir(), '.appium'); /** * Basename of extension manifest file. * @type {string} */ export const MANIFEST_BASENAME = 'extensions.yaml'; /** * Relative path to extension manifest file from `APPIUM_HOME`. * @type {string} */ export const MANIFEST_RELATIVE_PATH = path.join( 'node_modules', '.cache', 'appium', MANIFEST_BASENAME ); /** * Resolves `true` if an `appium` dependency can be found somewhere in the given `cwd`. * * @param {string} cwd * @returns {Promise<boolean>} */ export async function hasAppiumDependency(cwd) { return Boolean(await findAppiumDependencyPackage(cwd)); } /** * Given `cwd`, use `npm` to find the closest package _or workspace root_, and return the path if the root depends upon `appium`. * * Looks at `dependencies` and `devDependencies` for `appium`. */ export const findAppiumDependencyPackage = _.memoize( /** * @param {string} [cwd] * @param {string|semver.Range} [acceptableVersionRange='>=2.0.0-beta'] The expected * semver-compatible range for the Appium dependency. Packages that have 'appium' dependency * not satisfying this range will be skipped. * @returns {Promise<string|undefined>} */ async function findAppiumDependencyPackage ( cwd = process.cwd(), acceptableVersionRange = '>=2.0.0-beta' ) { /** * Tries to read `package.json` in `root` and resolves the identity if it depends on `appium`; * otherwise resolves `undefined`. * @param {string} root * @returns {Promise<string|undefined>} */ const readPkg = async (root) => { try { const pkg = await readPackageInDir(root); const version = semver.minVersion(String( pkg?.dependencies?.appium ?? pkg?.devDependencies?.appium ?? pkg?.peerDependencies?.appium )); return version && semver.satisfies(version, acceptableVersionRange) ? root : undefined; } catch {} }; let currentDir = path.resolve(cwd); let isAtFsRoot = false; while (!isAtFsRoot) { const result = await readPkg(currentDir); if (result) { return result; } currentDir = path.dirname(currentDir); isAtFsRoot = currentDir.length <= path.dirname(currentDir).length; } } ); /** * Read a `package.json` in dir `cwd`. If none found, return `undefined`. */ export const readPackageInDir = _.memoize( /** * * @param {string} cwd - Directory ostensibly having a `package.json` * @returns {Promise<import('read-pkg').NormalizedPackageJson|undefined>} */ async function _readPackageInDir(cwd) { return await readPkg({cwd, normalize: true}); } ); /** * Determines location of Appium's "home" dir * * - If `APPIUM_HOME` is set in the environment, use that * - If we find a `package.json` in or above `cwd` and it has an `appium` dependency, use that. * * All returned paths will be absolute. */ export const resolveAppiumHome = _.memoize( /** * @param {string} [cwd] - Current working directory. _Must_ be absolute, if specified. * @returns {Promise<string>} */ async function _resolveAppiumHome(cwd = process.cwd()) { if (!path.isAbsolute(cwd)) { throw new TypeError('`cwd` parameter must be an absolute path'); } if (process.env.APPIUM_HOME) { return path.resolve(cwd, process.env.APPIUM_HOME); } return await findAppiumDependencyPackage(cwd) ?? DEFAULT_APPIUM_HOME; } ); /** * Figure out manifest path based on `appiumHome`. * * The assumption is that, if `appiumHome` has been provided, it was resolved via {@link resolveAppiumHome `resolveAppiumHome()`}! If unsure, * don't pass a parameter and let `resolveAppiumHome()` handle it. */ export const resolveManifestPath = _.memoize( /** * @param {string} [appiumHome] - Appium home directory * @returns {Promise<string>} */ async function _resolveManifestPath(appiumHome) { // can you "await" in a default parameter? is that a good idea? appiumHome = appiumHome ?? (await resolveAppiumHome()); return path.join(appiumHome, MANIFEST_RELATIVE_PATH); } ); /** * @typedef {import('read-pkg').NormalizedPackageJson} NormalizedPackageJson */