Qaskills Selenide Testing
Expert-level Selenide UI testing skill for Java applications. Covers concise fluent API, automatic waits, smart selectors, collections, Page Objects, and integration with JUnit 5 and Gradle/Maven builds.
install
source · Clone the upstream repo
git clone https://github.com/PramodDutta/qaskills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/PramodDutta/qaskills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/seed-skills/selenide-testing" ~/.claude/skills/pramoddutta-qaskills-selenide-testing && rm -rf "$T"
manifest:
seed-skills/selenide-testing/SKILL.mdsource content
Selenide Testing Skill
You are an expert QA automation engineer specializing in Selenide UI testing for Java applications. When the user asks you to write, review, or debug Selenide tests, follow these detailed instructions.
Core Principles
- Concise fluent API -- Use Selenide's
and$
shortcuts instead of verbose Selenium WebDriver calls. Selenide wraps Selenium to provide a cleaner, more readable API.$$ - Automatic waits -- Selenide waits for elements automatically. Never add
or explicit waits unless absolutely necessary.Thread.sleep() - Smart selectors -- Prefer
attributes, then CSS selectors. Avoid XPath unless the DOM structure requires it.data-testid - Fail-fast assertions -- Use
,shouldBe
,shouldHave
conditions that produce clear error messages with screenshots on failure.shouldNot - Test isolation -- Each test must be independent. Use
to set up clean state. Never rely on test execution order.@BeforeEach
Project Structure
Always organize Selenide projects with this structure:
src/ test/ java/ com/example/ tests/ LoginTest.java DashboardTest.java CheckoutTest.java pages/ LoginPage.java DashboardPage.java BasePage.java config/ TestConfig.java data/ TestDataFactory.java utils/ TestHelpers.java resources/ selenide.properties build.gradle # or pom.xml
Setup
Maven
<dependencies> <dependency> <groupId>com.codeborne</groupId> <artifactId>selenide</artifactId> <version>7.2.0</version> <scope>test</scope> </dependency> <dependency> <groupId>org.junit.jupiter</groupId> <artifactId>junit-jupiter</artifactId> <version>5.10.2</version> <scope>test</scope> </dependency> </dependencies>
Gradle
dependencies { testImplementation 'com.codeborne:selenide:7.2.0' testImplementation 'org.junit.jupiter:junit-jupiter:5.10.2' }
Configuration (selenide.properties)
selenide.browser=chrome selenide.baseUrl=http://localhost:3000 selenide.timeout=10000 selenide.screenshots=true selenide.savePageSource=false selenide.headless=false selenide.pageLoadTimeout=30000
Basic Test Patterns
Login Test
import com.codeborne.selenide.*; import static com.codeborne.selenide.Selenide.*; import static com.codeborne.selenide.Condition.*; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.BeforeEach; class LoginTest { @BeforeEach void setUp() { Configuration.baseUrl = "http://localhost:3000"; Configuration.browser = "chrome"; Configuration.timeout = 10000; } @Test void loginWithValidCredentials() { open("/login"); $("#email").setValue("user@test.com"); $("#password").setValue("password123"); $("button[type='submit']").click(); $(".dashboard").shouldBe(visible); $(".welcome-message").shouldHave(text("Welcome")); } @Test void loginShowsErrorForInvalidCredentials() { open("/login"); $("#email").setValue("wrong@test.com"); $("#password").setValue("wrong"); $("button[type='submit']").click(); $(".error-message").shouldBe(visible) .shouldHave(text("Invalid credentials")); } @Test void loginRequiresEmail() { open("/login"); $("#password").setValue("password123"); $("button[type='submit']").click(); $("#email-error").shouldBe(visible); } }
Selectors Reference
// CSS selectors (preferred) $("[data-testid='login-btn']") // data-testid (best practice) $("css-selector") // Generic CSS $("#email") // By ID $(".submit-button") // By class // Text-based selectors $(byText("Login")) // Exact text match $(withText("Welc")) // Contains text $(byTitle("Submit Form")) // By title attribute // Attribute selectors $(byId("email")) // By ID $(byName("password")) // By name $(byAttribute("role", "button")) // By any attribute // XPath (avoid when possible) $(byXpath("//button[@type='submit']")) // Collections $$("li").shouldHave(size(5)); $$("li").first().shouldHave(text("Item 1")); $$("li").last().shouldHave(text("Item 5")); $$("li").filterBy(text("Active")).shouldHave(size(2)); $$("li").excludeWith(cssClass("disabled")).shouldHave(size(3)); $$("tr").findBy(text("Alice")).shouldBe(visible);
Conditions Reference
// Visibility element.shouldBe(visible); element.shouldBe(hidden); element.shouldNotBe(visible); element.shouldBe(exist); element.shouldNot(exist); // State element.shouldBe(enabled); element.shouldBe(disabled); element.shouldBe(readonly); element.shouldBe(focused); element.shouldBe(selected); element.shouldBe(checked); // Text and values element.shouldHave(text("expected")); element.shouldHave(exactText("Exact Match")); element.shouldHave(textCaseSensitive("CaseSensitive")); element.shouldHave(value("input value")); element.shouldHave(exactValue("exact input")); // Attributes and CSS element.shouldHave(attribute("href", "/link")); element.shouldHave(attribute("data-state", "active")); element.shouldHave(cssClass("active")); element.shouldHave(cssValue("color", "rgb(255, 0, 0)"));
Page Object Model
Base Page
import com.codeborne.selenide.SelenideElement; import static com.codeborne.selenide.Selenide.*; public abstract class BasePage { public abstract SelenideElement rootElement(); public boolean isDisplayed() { return rootElement().isDisplayed(); } protected void navigateTo(String path) { open(path); } }
Login Page
import com.codeborne.selenide.SelenideElement; import static com.codeborne.selenide.Selenide.*; import static com.codeborne.selenide.Condition.*; public class LoginPage extends BasePage { private final SelenideElement emailField = $("#email"); private final SelenideElement passwordField = $("#password"); private final SelenideElement submitButton = $("[data-testid='login-submit']"); private final SelenideElement errorMessage = $(".error-message"); private final SelenideElement forgotPasswordLink = $("a[href='/forgot-password']"); @Override public SelenideElement rootElement() { return $("[data-testid='login-form']"); } public LoginPage open() { navigateTo("/login"); rootElement().shouldBe(visible); return this; } public DashboardPage loginAs(String email, String password) { emailField.setValue(email); passwordField.setValue(password); submitButton.click(); return new DashboardPage(); } public LoginPage loginExpectingError(String email, String password) { emailField.setValue(email); passwordField.setValue(password); submitButton.click(); errorMessage.shouldBe(visible); return this; } public String getErrorMessage() { return errorMessage.getText(); } }
Test Using Page Objects
import org.junit.jupiter.api.Test; import static com.codeborne.selenide.Condition.*; class LoginPageTest { private final LoginPage loginPage = new LoginPage(); @Test void successfulLogin() { loginPage.open() .loginAs("user@test.com", "password123"); new DashboardPage().welcomeMessage() .shouldHave(text("Welcome")); } @Test void invalidLoginShowsError() { loginPage.open() .loginExpectingError("bad@test.com", "wrong"); assert loginPage.getErrorMessage().contains("Invalid"); } }
Collections and Iteration
import static com.codeborne.selenide.CollectionCondition.*; import static com.codeborne.selenide.Selenide.$$; // Size assertions $$(".todo-item").shouldHave(size(5)); $$(".todo-item").shouldHave(sizeGreaterThan(3)); $$(".todo-item").shouldHave(sizeGreaterThanOrEqual(5)); $$(".todo-item").shouldHave(sizeLessThan(10)); // Text assertions on collections $$(".menu-item").shouldHave(texts("Home", "About", "Contact")); $$(".menu-item").shouldHave(exactTexts("Home", "About", "Contact")); $$(".menu-item").shouldHave(textsInAnyOrder("Contact", "Home", "About")); // Filtering and finding $$("tr.user-row") .filterBy(text("Active")) .shouldHave(size(3)); $$("tr.user-row") .findBy(text("Alice")) .find(".delete-btn") .click(); // Iterating $$(".product-card").forEach(card -> { card.find(".price").shouldBe(visible); card.find(".title").shouldNotBe(empty); }); // First / last / get $$(".items").first().shouldHave(text("First")); $$(".items").last().shouldHave(text("Last")); $$(".items").get(2).shouldHave(text("Third"));
File Upload and Download
// Upload $("input[type='file']").uploadFile(new File("src/test/resources/test.pdf")); $("input[type='file']").uploadFromClasspath("test.pdf"); // Download File file = $("a.download-link").download(); assertThat(file.getName()).isEqualTo("report.pdf"); assertThat(file.length()).isGreaterThan(0);
Working with Frames and Windows
// Frames switchTo().frame("iframe-name"); $(".inside-frame").shouldBe(visible); switchTo().defaultContent(); // New windows String originalWindow = WebDriverRunner.getWebDriver().getWindowHandle(); $("a[target='_blank']").click(); switchTo().window(1); $(".new-window-content").shouldBe(visible); switchTo().window(originalWindow);
JavaScript Execution
// Execute JavaScript executeJavaScript("window.scrollTo(0, document.body.scrollHeight)"); long scrollHeight = executeJavaScript("return document.body.scrollHeight"); // Execute on element executeJavaScript("arguments[0].click()", $(".hidden-button")); executeJavaScript("arguments[0].scrollIntoView(true)", $(".target-element"));
Configuration Patterns
import com.codeborne.selenide.Configuration; // Programmatic configuration Configuration.browser = "chrome"; Configuration.baseUrl = "http://localhost:3000"; Configuration.timeout = 10000; Configuration.headless = true; Configuration.browserSize = "1920x1080"; Configuration.screenshots = true; Configuration.savePageSource = false; Configuration.reportsFolder = "build/reports/tests"; Configuration.downloadsFolder = "build/downloads"; // Remote WebDriver configuration Configuration.remote = "http://selenium-hub:4444/wd/hub"; Configuration.browserCapabilities = new DesiredCapabilities(); Configuration.browserCapabilities.setCapability("browserName", "chrome");
CI/CD Integration
GitHub Actions
name: Selenide Tests on: [push, pull_request] jobs: test: runs-on: ubuntu-latest steps: - uses: actions/checkout@v4 - uses: actions/setup-java@v4 with: java-version: '17' distribution: 'temurin' - name: Run tests run: ./gradlew test - name: Upload reports if: always() uses: actions/upload-artifact@v4 with: name: selenide-reports path: build/reports/tests/
Best Practices
- Use data-testid selectors -- Add
attributes to elements specifically for testing. They survive CSS refactors and are explicit about their purpose.data-testid - Let Selenide handle waits -- Never use
. Selenide's built-in implicit waits handle dynamic content. Only increaseThread.sleep()
if you have genuinely slow pages.Configuration.timeout - One assertion per concept -- Group related assertions but keep each test focused on one behavior. Use descriptive test method names.
- Page Object encapsulation -- Never expose
fields publicly. Instead, expose action methods (SelenideElement
,loginAs
) that return the next page object.addToCart - Use collections wisely -- Use
for lists and tables. Filter with$$()
andfilterBy
instead of iterating manually.findBy - Configure in properties file -- Use
for environment-specific config. Override in CI with system properties (selenide.properties
).-Dselenide.headless=true - Screenshot on failure -- Selenide captures screenshots automatically on failure. Configure
to a CI-accessible location.reportsFolder - Clean browser state -- Use
with@BeforeEach
or open a fresh browser per test for true isolation.Selenide.clearBrowserCookies() - Avoid over-abstracting -- Page Objects should match user mental models. Don't create deep inheritance hierarchies or overly generic helpers.
- Run headless in CI -- Set
in CI to avoid display server dependencies and speed up execution.Configuration.headless = true
Anti-Patterns
- Thread.sleep() for synchronization -- Never use
. Selenide's auto-waiting handles element readiness. If an element takes long, increase the timeout or check if the page has a loading indicator.Thread.sleep() - XPath as default selector strategy -- XPath is brittle and hard to read. Use CSS selectors or Selenide's text-based finders instead.
- Test interdependency -- Tests that depend on each other or must run in a specific order will cause cascading failures and are impossible to run in parallel.
- Hardcoded URLs -- Never hardcode full URLs in tests. Use
and relative paths.Configuration.baseUrl - Ignoring collection assertions -- Using
without first asserting the collection size leads to cryptic index errors.$$().get(0).shouldHave(text("x")) - Giant test methods -- Tests with 50+ lines of actions and assertions are unreadable. Break them into smaller focused tests or extract helper methods.
- Testing implementation details -- Don't assert on CSS classes for styling or internal DOM structure. Assert on user-visible behavior.
- Shared mutable state between tests -- Static variables or class fields that accumulate state across tests cause flaky results.
- Catching exceptions in tests -- Don't wrap Selenide calls in try/catch. Let assertions fail naturally with Selenide's clear error messages and screenshots.
- Skipping Page Objects for simple tests -- Even simple tests benefit from Page Objects. Inline selectors scattered across test files become maintenance nightmares.
Run Commands
# Maven mvn test mvn test -Dselenide.headless=true mvn test -Dtest=LoginTest mvn test -Dselenide.browser=firefox # Gradle ./gradlew test ./gradlew test --tests "com.example.tests.LoginTest" ./gradlew test -Dselenide.headless=true