testium-driver-wd
Version:
Testium backend powered by wd
508 lines (454 loc) • 13.3 kB
JavaScript
/*
* Copyright (c) 2015, Groupon, Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* Neither the name of GROUPON nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
;
const assert = require('assert');
const asserters = require('./_asserters');
// Our own version of wd's waitForElement.
// wd's version of waitFor* has the unfortunate habit of generating really
// unhelpful error messages.
// eslint-disable-next-line no-underscore-dangle
function _waitForElement(browser, selector, asserter, timeout, pollFreq) {
/* eslint no-use-before-define:0 */
timeout = timeout || 2000;
pollFreq = pollFreq || 50;
const endTime = Date.now() + timeout;
async function attempt() {
const element = await browser.elementByCssSelectorOrNull(selector);
try {
await asserter.assert(element);
} catch (e) {
return retry(e);
}
return element;
}
function retry(error) {
if (!error.retriable) {
throw error;
}
if (Date.now() >= endTime) {
error.message = `Timeout (${timeout}ms): ${error.message}`;
throw error;
}
return new Promise(resolve => setTimeout(resolve, pollFreq)).then(attempt);
}
return attempt();
}
/**
* @param {string} selector
* @return {Promise<Element|null>}
*
* @example browser.getElementOrNull('#foo')
*/
function getElementOrNull(selector) {
return this.elementByCssSelectorOrNull(selector);
}
/**
* @param {string} selector
* @return {Promise<Element>}
*
* @example browser.getElement('#foo')
*/
async function getElement(selector) {
const element = await this.elementByCssSelectorOrNull(selector);
if (element === null) {
throw new Error(`Element not found for selector: ${selector}`);
}
return element;
}
/**
* @param {string} selector
* @return {Promise<Element[]>}
*
* @example browser.getElements('#foo > li')
*/
function getElements(selector) {
return this.elementsByCssSelector(selector);
}
/**
* @param {string} selector
* @return {Promise<void>}
*
* @example browser.clickOn('#foo')
*/
async function clickOn(selector) {
const elements = await this.getElements(selector);
if (elements.length === 1) {
elements[0].click();
} else if (elements.length > 1) {
throw new Error(
`selector "${selector}" matched more than 1 element. Use .clickOnAll() or a more specific selector instead.`
);
} else {
throw new Error(`selector "${selector}" matched no element.`);
}
}
/**
* @param {string} selector
* @return {Promise<void>}
*
* @example browser.clickOnAll('#foo a')
*/
async function clickOnAll(selector) {
const elementList = await this.getElements(selector);
await Promise.all(elementList.map(elem => elem.click()));
}
/**
*
* @param {import('wd').Asserter} asserter
* @private
*/
// eslint-disable-next-line no-underscore-dangle
function _assert(asserter) {
return asserter.assert(this);
}
/**
* @param {string} selector
* @param {{[key:string]: Promise<function>}} asserter
* @return {Promise<Element>}
* @private
*/
async function assertElement(selector, asserter) {
assert.strictEqual(typeof selector, 'string', 'selector should be a string');
const element = await this.elementsByCssSelector(selector).then(elements => {
switch (elements.length) {
case 0:
throw new Error(`Element not found for selector: ${selector}`);
case 1:
return elements[0];
default:
throw new Error(
`Selector ${selector} has ${elements.length} hits on the page, assertions require unique elements`
);
}
});
await asserter.assert(element);
return element;
}
/**
* @param {string} selector
* @return {Promise<Element>}
*
* @example browser.assertElementDisplayed('#foo')
*/
async function assertElementIsDisplayed(selector) {
return await this.assertElement(
selector,
asserters.isDisplayed(true, selector)
);
}
/**
* @param {string} selector
* @return {Promise<Element>}
*
* @example browser.assertElementNotDisplayed('#foo')
*/
async function assertElementNotDisplayed(selector) {
const element = await this.elementByCssSelectorOrNull(selector);
await asserters.isDisplayed(false, selector).assert(element);
return element;
}
/**
* @param {string} selector
* @return {Promise<Element>}
*
* @example browser.assertElementExists('#foo')
*/
async function assertElementExists(selector) {
const element = await this.elementByCssSelectorOrNull(selector);
await asserters.exists(true, selector).assert(element);
return element;
}
/**
* @param {string} selector
* @return {Promise<null>}
*
* @example browser.assertElementDoesntExist('#foo')
*/
async function assertElementDoesntExist(selector) {
const element = await this.elementByCssSelectorOrNull(selector);
await asserters.exists(false, selector).assert(element);
return null;
}
/**
* @param {string} selector
* @param {string|RegExp} textOrRegExp
* @return {Promise<Element>}
*
* @example browser.assertElementHasText('#foo', 'bar')
* @example browser.assertElementHasText('#foo', /bar/)
*/
function assertElementHasText(selector, textOrRegExp) {
return this.assertElement(
selector,
asserters.fuzzyString('text', textOrRegExp, true, selector)
);
}
/**
* @param {string} selector
* @param {string|RegExp} textOrRegExp
* @return {Promise<Element>}
*
* @example browser.assertElementLacksText('#foo', 'bar')
* @example browser.assertElementLacksText('#foo', /bar/)
*/
function assertElementLacksText(selector, textOrRegExp) {
return this.assertElement(
selector,
asserters.fuzzyString('text', textOrRegExp, false, selector)
);
}
/**
* @param {string} selector
* @param {string|RegExp} textOrRegExp
* @return {Promise<Element>}
*
* @example browser.assertElementHasValue('#foo input', 'bar')
* @example browser.assertElementHasValue('#foo input', /bar/)
*/
function assertElementHasValue(selector, textOrRegExp) {
return this.assertElement(
selector,
asserters.fuzzyString('getValue', textOrRegExp, true, selector)
);
}
/**
* @param {string} selector
* @param {string|RegExp} textOrRegExp
* @return {Promise<Element>}
*
* @example browser.assertElementLacksValue('#foo input', 'bar')
* @example browser.assertElementLacksValue('#foo input', /bar/)
*/
function assertElementLacksValue(selector, textOrRegExp) {
return this.assertElement(
selector,
asserters.fuzzyString('getValue', textOrRegExp, false, selector)
);
}
/**
* @param {string} selector
* @param {string} attribute
* @return {Promise<Element>}
*
* @example browser.assertElementHasAttribute('#foo', 'data-test')
*/
function assertElementHasAttribute(selector, attribute) {
assert.strictEqual(
typeof attribute,
'string',
`attribute must by of type string`
);
function assertAttribute(element) {
return element.getAttribute(attribute).then(actual => {
assert.notStrictEqual(actual, null);
});
}
return this.assertElement(selector, { assert: assertAttribute });
}
/**
* @param {string} selector
* @param {Record<string, any>} attributes
* @return {Promise<Element>}
*
* @example browser.assertElementHasAttributes('#foo', { alt: 'I was here', 'data-test': 'test' })
*/
function assertElementHasAttributes(selector, attributes) {
assert.ok(
/^\[object\sObject]$/.test(Object.prototype.toString.call(attributes)),
'attributes should be an object'
);
function assertAttributes(element) {
const assertions = Object.entries(attributes).map(([attr, expected]) => {
return element.getAttribute(attr).then(actual => {
assert.strictEqual(
actual,
expected,
`Assertion failed: attribute "${attr}"\nExpected: "${expected}"\nActually: ${actual}`
);
});
});
return Promise.all(assertions);
}
return this.assertElement(selector, { assert: assertAttributes });
}
/**
* @param {string} selector
* @param {string} attribute
* @return {Promise<Element>}
*
* @example browser.assertElementLacksAttribute('#foo', 'data-test')
*/
function assertElementLacksAttribute(selector, attribute) {
assert.strictEqual(
typeof attribute,
'string',
`attribute must by of type string`
);
return this.assertElementLacksAttributes(selector, [attribute]);
}
/**
* @param {string} selector
* @param {string[]} attributes
* @return {Promise<Element>}
*
* @example browser.assertElementLacksAttributes('#foo', ['alt', 'data-test'])
*/
function assertElementLacksAttributes(selector, attributes) {
assert.ok(
Array.isArray(attributes),
'attributes should be an Array or strings'
);
function assertAttributes(element) {
const assertions = attributes.map(attr => {
return element.getAttribute(attr).then(actual => {
assert.strictEqual(
actual,
null,
`Assertion failed: attribute "${attr}" exists`
);
});
});
return Promise.all(assertions);
}
return this.assertElement(selector, { assert: assertAttributes });
}
/**
*
* @param {string} selector
* @param {string|{min?: number, eq?: number, max?: number}} numOrObj
* @return {Promise<Element[]>}
*
* @example browser.assertElementsNumber('.el', 3)
* @example browser.assertElementsNumber('.el', { eq: 3 })
* @example browser.assertElementsNumber('.el', { min: 1 })
* @example browser.assertElementsNumber('.el', { max: 10 })
* @example browser.assertElementsNumber('.el', { min: 1, max: 10 })
*/
async function assertElementsNumber(selector, numOrObj) {
const elements = await this.elementsByCssSelector(selector);
await asserters.elementsNumber(numOrObj, selector).assert(elements);
return elements;
}
/**
* @param {string} selector
* @param {number?} timeout
* @return {Promise<Element>}
*
* @example browser.waitForElementDisplayed('#foo')
*/
function waitForElementDisplayed(selector, timeout) {
return _waitForElement(
this,
selector,
asserters.isDisplayed(true, selector),
timeout
);
}
/**
* @param {string} selector
* @param {number?} timeout
* @return {Promise<Element>}
*
* @example browser.waitForElementNotDisplayed('#foo')
*/
function waitForElementNotDisplayed(selector, timeout) {
return _waitForElement(
this,
selector,
asserters.isDisplayed(false, selector),
timeout
);
}
/**
* @param {string} selector
* @param {number?} timeout
* @param {number?} pollFreq
* @return {Promise<Element>}
*
* @example browser.waitForElementExist('#foo')
* @example browser.waitForElementExist('#foo', 5000)
* @example browser.waitForElementExist('#foo', 5000, 300)
*/
function waitForElementExist(selector, timeout, pollFreq) {
return _waitForElement(
this,
selector,
asserters.exists(true, selector),
timeout,
pollFreq
);
}
/**
* @param {string} selector
* @param {number?} timeout
* @param {number?} pollFreq
* @return {Promise<null>}
*
* @example browser.waitForElementNotExist('#foo')
* @example browser.waitForElementNotExist('#foo', 5000)
* @example browser.waitForElementNotExist('#foo', 5000, 300)
*/
function waitForElementNotExist(selector, timeout, pollFreq) {
return _waitForElement(
this,
selector,
asserters.exists(false, selector),
timeout,
pollFreq
);
}
module.exports = {
assert: _assert,
assertElement,
assertElementHasText,
assertElementLacksText,
assertElementHasValue,
assertElementLacksValue,
assertElementHasAttribute,
assertElementHasAttributes,
assertElementLacksAttribute,
assertElementLacksAttributes,
assertElementsNumber,
assertElementExists,
assertElementDoesntExist,
assertElementIsDisplayed,
assertElementNotDisplayed,
clickOn,
clickOnAll,
getElement,
getElements,
getElementOrNull,
waitForElementDisplayed,
waitForElementNotDisplayed,
waitForElementExist,
waitForElementNotExist,
};