我正在尝试使现有的解决方案可以在任何平台上运行。制作扩展 DriverSource 类的自定义驱动程序很容易,但我仍然必须解决不同的页面对象注入(例如在移动设备上运行具有不同的布局)。为此,我想使用 Google Guice
我将分享一个没有 DI 的解决方案,它工作正常,但应该有更好的方法。
public interface GooglePageInterface {
public void enter_keywords(String keyword);
public void lookup_terms();
public void navigateToHomePage();
}
(您也可以使用抽象类而不是接口,因为移动设备和桌面设备的大多数方法可能相同)
以及接口的两种实现:
public class GoogleMobilePage extends PageObject implements GooglePageInterface {
@FindBy(name = "q")
private WebElementFacade searchTerms;
@FindBy(id = "tsbb")
private WebElementFacade lookupButton;
@Override
public void enter_keywords(String keyword) {
element(searchTerms).waitUntilVisible();
searchTerms.sendKeys(keyword);
}
@Override
public void lookup_terms() {
lookupButton.click();
}
// demo purpose- there are better ways for this
@Override
public void navigateToHomePage() {
getDriver().get("https://www.google.ro/");
}
}
和
public class GoogleDesktopPage extends PageObject implements GooglePageInterface {
@FindBy(name = "q")
private WebElementFacade searchTerms;
@FindBy(css = "button[name='btnG']")
private WebElementFacade lookupButton;
@Override
public void enter_keywords(String keyword) {
element(searchTerms).waitUntilVisible();
searchTerms.sendKeys(keyword);
}
@Override
public void lookup_terms() {
lookupButton.click();
}
@Override
public void navigateToHomePage() {
System.out.println("You are in GoogleDesktopPage")
getDriver().get("https://www.google.ro/");
}
AbstractSteps 将根据 Cmd 中给出的系统属性来决定实现
public class AbstractSteps extends ScenarioSteps {
private static final long serialVersionUID = 1L;
public GooglePageInterface getDictionaryPage() {
switch (System.getProperty("runPlatform")) {
case "desktop":
return getPages().currentPageAt(GoogleDesktopPage.class);
case "mobile":
return getPages().currentPageAt(GoogleMobilePage.class);
default:
return null;
}
}
}
EndUserStep 将扩展 AbstractSteps 并使用方法
public class EndUserSteps extends AbstractSteps{
private static final long serialVersionUID = 1L;
@Step
public void enters(String keyword) {
getDictionaryPage().enter_keywords(keyword);
}
@Step
public void starts_search() {
getDictionaryPage().lookup_terms();
}
@Step
public void navigateToHomePage() {
getDictionaryPage().navigateToHomePage();
}
@Step
public void looks_for(String term) {
enters(term);
starts_search();
}
}
和一个基本的测试:
@RunWith(SerenityRunner.class)
public class SearchByKeywordStory {
@Managed(uniqueSession = true)
public WebDriver webdriver;
@Steps
public EndUserSteps endUserSteps;
@Test
public void searching_by_keyword_apple_should_display_the_corresponding_article() {
endUserSteps.navigateToHomePage();
endUserSteps.looks_for("something");
}
}
我使用以下命令运行测试:
mvn test -Dtest=SearchByKeywordStory -Dwebdriver.driver=provided -DrunPlatform=mobile -DrunEnv=staging-env verify,其中提供的是移动设备的自定义驱动程序:
public class CustomDriver implements DriverSource {
@Override
public WebDriver newDriver() {
return setChromeMobile();
}
@Override
public boolean takesScreenshots() {
return true;
}
private WebDriver setChromeMobile() {
Map<String, Object> deviceMetrics = new HashMap<String, Object>();
//landscape
deviceMetrics.put("width", 732);
deviceMetrics.put("height", 412);
deviceMetrics.put("pixelRatio", 3.0);
Map<String, Object> mobileEmulation = new HashMap<String, Object>();
mobileEmulation.put("deviceMetrics", deviceMetrics);
mobileEmulation.put("userAgent", "Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/55.0.2883.87 Mobile Safari/537.36");
Map<String, Object> chromeOptions = new HashMap<String, Object>();
chromeOptions.put("mobileEmulation", mobileEmulation);
DesiredCapabilities capabilities = DesiredCapabilities.chrome();
capabilities.setCapability(ChromeOptions.CAPABILITY, chromeOptions);
return new ChromeDriver(capabilities);
}
}
在 serenity.properties
webdriver.driver=provided
webdriver.provided.type = mydriver
webdriver.provided.mydriver = com.tools.customerDrivers.CustomDriver
thucydides.driver.capabilities = mydriver
完成,它工作!
但正如我之前所说,我想尝试使用 Guice 完成这项工作,但我遇到了问题,因为 Serenity 使用 Google Guice 实例化驱动程序和步骤。
页面接口和页面实现保持不变。
步骤:
public class EndUserSteps2 extends ScenarioSteps{
private static final long serialVersionUID = 1L;
@Inject
GooglePageInterface dictionaryPage;
@Step
public void enters(String keyword) {
dictionaryPage.enter_keywords(keyword);
}
@Step
public void starts_search() {
dictionaryPage.lookup_terms();
}
@Step
public void navigateToHomePage() {
dictionaryPage.navigateToHomePage();
}
@Step
public void looks_for(String term) {
enters(term);
starts_search();
}
我创建了一个 BaseTest,其中创建了 injecor(这可以通过不同的方式完成) 完成绑定的模块 -bind(GooglePageInterface.class).to(GoogleDesktopPage.class)- 可以在不同的类中
@RunWith(SerenityRunner.class)
public class BaseTest {
protected Injector injector = Guice.createInjector(new AbstractModule() {
@Override
protected void configure() {
bind(GooglePageInterface.class).to(GoogleDesktopPage.class);
}
});
@Before
public void setup() {
injector.injectMembers(this);
}
}
和测试:
@RunWith(SerenityRunner.class)
public class SearchByKeywordStory2 extends BaseTest {
@Managed(uniqueSession = true)
public WebDriver webdriver;
//here is the issue
@Inject
@Steps
public EndUserSteps2 endUserSteps;
@Test
public void searching_by_keyword_apple_should_display_the_corresponding_article() {
endUserSteps.navigateToHomePage();
endUserSteps.looks_for("something");
}
}
测试开始,但驱动程序未传递给 GoogleDesktopPage.class,因此测试失败,但该类被注入,因为来自 navigateToHomePage() 的消息“您在 GoogleDesktopPage”被打印出来
我知道 Serenity 使用注释注入 webdriver 和步骤,我有点卡在这里
如果有人能以某种方式帮助我完成这项工作,我将不胜感激(可能是与 DI 不同的方法,因为这似乎不是一个好方法)