Skip to content

Latest commit

 

History

History
168 lines (144 loc) · 8.17 KB

e2e.md

File metadata and controls

168 lines (144 loc) · 8.17 KB

End to end test framework

Grafana Labs uses a minimal home grown solution built on top of Cypress for our end to end (e2e) tests.

Basic concepts

Here is a good introduction to e2e best practices: https://martinfowler.com/bliki/PageObject.html.

  • Selector: A unique identifier that is used from the e2e framework to retrieve an element from the Browser
  • Page: An abstraction for an object that contains one or more Selectors
  • Flow: An abstraction that contains a sequence of actions on one or more Pages that can be reused and shared between tests

Basic example

Let's start with a simple example with a single selector. For simplicity, all examples are in JSX.

In our example app, we have an input that we want to type some text into during our e2e test.

<div>
    <input type="text" className="gf-form-input login-form-input"/>
</div>

We could define a selector using JQuery type selectors with a string like '.gf-form-input.login-form-input' but that would be brittle as style changes occur frequently. Furthermore there is nothing that signals to future developers that this input is part of an e2e test.

At Grafana, we use aria-label as our preferred way of defining selectors instead of data-* attributes. This also aids in accessibility. Let's add a descriptive aria-label to our simple example.

<div>
    <input type="text" className="gf-form-input login-form-input" aria-label="Username input field"/>
</div>

Now that we added the aria-label we suddenly get more information about this particular field. It's an input field that represents a username, but there it's still not really signaling that it's part of an e2e test.

The next step is to create a Page representation in our e2e test framework to glue the test with the real implementation using the pageFactory function. For that function we can supply a url and selectors like in the example below:

export const Login = pageFactory({
  url: '/login', // used when called from Login.visit()
  selectors: {
    username: 'Username input field', // used when called from Login.username().type('Hello World')
  },
});

The next step is to add the Login page to the exported const Pages in packages/grafana-e2e/src/pages/index.ts so that it appears when we type e2e.pages in our IDE.

export const Pages = {
  Login,
  ...,
  ...,
  ...,
};

Now that we have a Page called Login in our Pages const we can use that to add a selector in our html like shown below and now this really signals to future developers that it is part of an e2e test.

<div>
    <input type="text" className="gf-form-input login-form-input" aria-label={e2e.pages.Login.selectors.username}/>
</div>

The last step in our example is to use our Login page as part of a test. The pageFactory function we used before gives us two things:

  • The url property is used whenever we call the visit function and is equivalent to the Cypress function cy.visit().

Best practice after calling visit is to always call should on a selector to prevent flaky tests when you try to access an element that isn't ready. For more information, refer to Commands vs. assertions.

  • Any defined selector in the selectors property can be accessed from the Login page by invoking it. This is equivalent to the result of the Cypress function cy.get(...).
describe('Login test', () => {
  it('Should pass', () => {
    e2e.pages.Login.visit();
    // To prevent flaky tests, always do a .should on any selector that you expect to be in the DOM.
    // Read more here: https://docs.cypress.io/guides/core-concepts/retry-ability.html#Commands-vs-assertions
    e2e.pages.Login.username().should('be.visible');
    e2e.pages.Login.username().type('admin');
  });
});

Advanced example

Let's take a look at an example that uses the same selector for multiple items in a list for instance. In this example app we have a list of data sources that we want to click on during an e2e test.

<ul>
  {dataSources.map(dataSource => (
    <li className="card-item-wrapper" key={dataSource.id}>
      <a className="card-item" href={`datasources/edit/${dataSource.id}`}>
        <div className="card-item-name">
          {dataSource.name}
        </div>
      </a>
    </li>
  ))}
</ul>

Just as before in the basic example we'll start by creating a page abstraction using the `pageFactory` function:
```typescript
export const DataSources = pageFactory({
  url: '/datasources',
  selectors: {
    dataSources: (dataSourceName: string) => `Data source list item ${dataSourceName}`,
  },
});

You might have noticed that instead of a simple string as the selector, we're using a function that takes a string parameter as an argument and returns a formatted string using the argument.

Just as before we need to add the DataSources page to the exported const Pages in packages/grafana-e2e/src/pages/index.ts.

The next step is to use the dataSources selector function as in our example below:

<ul>
  {dataSources.map(dataSource => (
    <li className="card-item-wrapper" key={dataSource.id}>
      <a className="card-item" href={`datasources/edit/${dataSource.id}`}>
        <div className="card-item-name" aria-label={e2e.pages.DataSources.selectors.dataSources(dataSource.name)}>
          {dataSource.name}
        </div>
      </a>
    </li>
  ))}
</ul>

When this list is rendered with the data sources with names A, B, C the resulting html would become:

<div class="card-item-name" aria-label="Data source list item A">
 A
</div>
...
<div class="card-item-name" aria-label="Data source list item B">
 B
</div>
...
<div class="card-item-name" aria-label="Data source list item C">
 C
</div>

Now we can write our test. The one thing that differs from the Basic example is that we pass in which data source we want to click on as an argument to the selector function:

Best practice after calling visit is to always call should on a selector to prevent flaky tests when you try to access an element that isn't ready. For more information, refer to Commands vs. assertions.

describe('List test', () => {
  it('Clicking on data source named B', () => {
    e2e.pages.DataSources.visit();
    // To prevent flaky tests, always do a .should on any selector that you expect to be in the DOM.
    // Read more here: https://docs.cypress.io/guides/core-concepts/retry-ability.html#Commands-vs-assertions
    e2e.pages.DataSources.dataSources('B').should('be.visible');
    e2e.pages.DataSources.dataSources('B').click();
  });
});

Debugging PhantomJS image rendering

Common Error

The most common error with PhantomJs image rendering is when a PR introduces an import that has functionality that's not supported by PhantomJs. To quickly identify which new import causes this you can use a tool like es-check.

  1. Run > npx es-check es5 './public/build/*.js'
  2. Check the output for files that break es5 compatibility.
  3. Lazy load the failing imports if possible.

Debugging

There is no easy or comprehensive way to debug PhantomJS smoke test (image rendering) failures. However, PhantomJS exposes remote debugging interface which can give you a sense of what is going wrong in the smoke test. Before performing the steps described below make sure your local Grafana instance is running:

  1. Go to tools/phantomjs directory
  2. Execute phantomjs binary against render.js file: ./phantomjs --remote-debugger-port=9009 --remote-debugger-autorun=yes ./render.js url="http://localhost:3000"
  3. In your browser navigate to http://localhost:9009/
  4. Select http://localhost:3000/login from the list. You will get access to Webkit's inspector to see the console's output from the smoke test.

The method described above is not perfect, but is helpful to evaluate smoke tests breaking due to bundle errors.