Automated UI testing and SAIL

The Skyve platform provides the ability to generate a suite of automated User Interface tests via the Skyve Abstract Interaction Language (SAIL).

SAIL is a high level language for describing interactions with applications, specified in a format which can be used to generate Selenese or Web Driver interactions.

SAIL is abstract in the sense that test cases are specified in terms of general user focused interactions like save or delete, rather than in terms which relate specifically to a particular renderer or user agent/device. This ensures that tests can be expressed in terms of the capability of the application, even where the specifics of how the UI is laid out or the specific UI mechanisms may be subject to change. Our intention is that the SAIL test case specification for an application can remain (as far as possible) independent of the particular rendering technology in use.

Because Skyve metadata is self-describing, and because Skyve user interfaces are consistent and predictable, Skyve can generate SAIL tests for many aspects of your application. The SAIL test cases can then be expanded, modified and combined to increase test coverage or to include highly specific test cases.

Once your SAIL test cases are specified, Skyve can then generate Selenese or Web Driver interactions.

The basic steps to generate automated UI tests are:

  1. Use the Skyve Control Panel (DevOps menu in Admin module) to generate SAIL (generated to the Results tab) Generate the SAIL
  2. Copy the generated SAIL from the Results tab, then review, modify and organise the SAIL test case descriptions Organise the SAIL
  3. Paste the resulting SAIL into the SAIL text area on the Control Panel SAIL tab then press Execute to generate Selenese or Web Driver test methods (generated to the Results tab) Generate Test class methods
  4. Copy the generated methods from the Results tab into your test classes as appropriate Organise test class methods

SAIL test generation builds on Skyve’s automated unit testing infrastructure, to be able to generate valid data for inputs according to the domain declaration.

NOTE Automated User Interface testing impacts the persisted data and content of your application, and does not use an in-memory database (like automated unit testing).

Generating SAIL

To access the DevOps menu in your Skyve application, ensure that your user has the DevOps role granted to them.

Steps to generate the SAIL test cases:

  • Click the Control Panel menu item and switch to the SAIL tab.
  • Specify the user credential which exemplifies the user profile for the suite of tests to be generated. (To generate SAIL appropriately, the generator needs to know which menu items and actions will be accessible during the test, and this is determined by the user profile.), e.g. myUser
  • Select the module to test, e.g. myModule
  • Select the UX/UI to test, e.g. external
  • Select the User Agent Type, e.g. Tablet
  • Select the Test Strategy, e.g. Verify
  • Select the Generation option, e.g. Menu SAIL

Generating SAIL test cases

Note that while the UX/UI and User Agent Type are specified, this is only so that the SAIL generation will include only menu items and actions which would be available to that combination - SAIL itself is independent of these.

Test strategies

Strategy Description Usage
Assert the test will be abandoned as soon as a test case fails used as the basis of go/no go decisions about whether the application is ready to progress (e.g. from Test to Production)
Verify the test will continue if a test case fails used by developers to analyse what further development/remediation activity is required
None not testing used for automated interactions which are not tests

Generation options

The Menu SAIL generation option will generate SAIL test cases for every menu option available to the specified user/UX/UI combination.

Module SAIL generation option will generate SAIL test cases for every document accessible by the user/UX/UI combination which could possibly be accessed, irrespective of the menu items declared in the module. This option may be useful where the system uses other menuing mechanisms not declared in the usual way.

Understanding SAIL

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<automation uxui="external" userAgentType="tablet" testStrategy="Assert" 
	xsi:schemaLocation="http://www.skyve.org/xml/sail ../../../skyve/schemas/sail.xsd" 
	xmlns="http://www.skyve.org/xml/sail" 
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
	
    <interaction name="Menu Document Numbers">
        <method>
            <navigateList document="DocumentNumber" module="admin"/>
            <listGridNew document="DocumentNumber" module="admin"/>
            <testDataEnter/>
            <save/>
            <testDataEnter/>
            <save/>
            <delete/>
        </method>
    </interaction>

The above example shows a typical SAIL interaction for testing Create-Read-Update-Delete capabilities, as follows:

  • navigate to the list for the DocumentNumber document in the admin module
  • select New from the list to create a new DocumentNumber
  • enter randomly generated (but valid) input data across all data entry widgets
  • save the DocumentNumber record
  • enter randomly generated (but valid) input data across all data entry widgets
  • save the DocumentNumber record
  • delete the DocumentNumber record

By starting with <listGridNew> and ending with <delete>, the interaction leaves the data store in the same state it was in prior to the test (provided the test completes successfully). This is not required for SAIL tests, but developers need to consider the impact of the tests they generate/construct and whether additional interactions may be required for setup and teardown.

The SAIL schema (sail.xsd) is included in the standard Skyve project distribution in the schemas folder.

Each SAIL interaction must be named and contain one or more method items.

Available method items include:

Item Description Example(s)
action perform a declared action <action name="MakePasswordChange"/>
cancel press the default cancel action button <cancel>
dataEnter enter data into a widget <dataEnter binding="url" value="skyve.org"/>
dataGridEdit enter a bound column for inline editing in a dataGrid (press the Edit button or right-click to Edit) for the collectin binding <dataGridEdit row="0" binding="websites" />
dataGridNew create a new element in the dataGrid collection binding (press the New button) <dataGridNew binding="websites"/>
dataGridRemove remove the element at row X from the dataGrid collection binding (press the Remove button) <dataGridRemove row="0" binding="websites"/>
dataGridSelect select the row X from the dataGrid collection binding <dataGridSelect row="0" binding="websites"/>
dataGridZoom zoom into row X from the dataGrid collection binding <dataGridSelect row="0" binding="websites"/>
delete delete the current record (implicit delete action) <delete/>
execute not yet implemented  
listGridNew create a new element from the listGrid (press the New button) <listGridNew module="admin" document="DocumentNumber" /> <listGridNew query="qDocumentNumber"/> <listGridNew model="myModel"/> <listGridNew module="admin" document="User" createView="true"/>
listGridSelect select row X from the listGrid <listGridSelect module="admin" document="User" row="0"/>
listGridZoom zoom into row X from the listGrid <listGridZoom module="admin" document="User" row="0"/>
lookupDescriptionAutoComplete autocomplete in the lookupDescription widget for binding using the search value <lookupDescriptionAutoComplete search="Elon Mu" binding="contact"/>
lookupDescriptionEdit edit the related record which is currently selected in the lookupDescription widget for the binding <lookupDescriptionEdit binding="contact"/>
lookupDescriptionNew create a new related record for the lookupDescription widget for the binding <lookupDescriptionNew binding="contact"/>
lookupDescriptionPick pick an item from the pick list of the lookupDescription widget for the binding <lookupDescriptionPick row="0" binding="contact"/>
navigateCalendar not yet implemented  
navigateEdit navigate to an Edit view for the specified module document <navigateEdit document="ChangePassword" module="admin"/>
navigateLink navigate to the link <navigateLink href="https://www.skyve.org"/>
navigateList navigate to the List view for the specified module document <navigateList document="Contact" module="admin"/>
navigateMap not yet implemented  
navigateTree navigate to the element in the tree view for the module document with the start and end <navigateTree module="staffManagement" document="OrganisationalStructure" startBinding="currentStaff" endBinding="currentStaffReport"/>
ok press the ok implicit action button <ok/>
remove press the remove implicit action button <remove/>
save press the save implicit action button <save/>
tabSelect select the specified tab from the tabPane <tabSelect path="Jobs"/>
testDataEnter select each data entry widget and enter valid random data (according to the attribute declaration of type, format and validator) <testDataEnter/>
testFailure signal the test has failed with message <testFailure message="The contact could not be added" />
testSuccess signal the test was successful <testSuccess/>
testValue test that the widget with binding has the specified value <testValue value="123.5" binding="total"/>
zoomOut press the zoomOut implicit action button <zoomOut/>

Using <testDataEnter/> for Happy paths

The <testDataEnter/> step is a high-level step which sets focus into each available data entry widget (in tab order) and enters randomly generated character sequences which are valid according to the attribute declaration of type, converter, format and validator.

This means that <testDataEnter/> effectively follows a Happy test path. For other paths, developers need to specify each data entry value using the <dataEnter/> step.

For example, if a view has 4 attributes - text, dateOnly, decimal2 and boolean - using <testDataEnter/> will exercise all four attributes in the order they appear in the view, and enter appropriate data for each, specifically:

  • text according to the specified attribute length and validator
  • dateOnly according to the declared converter (or default customer date converter if the attribute does not declare the converter)
  • decimal2 according to the declared converter (if one is declared)
  • boolean as true, false or null (if the attribute is not required)

(This list is not exhaustive - refer to the DataBuilder() description in Automated Unit Testing

Generating Selenese or Web Driver methods

Steps to generate methods:

  • With the settings as specified above for generating SAIL, paste the contents of your SAIL xml file into the SAIL text area on the Control Panel
  • Specify the desired executor (either PrimeFaces Inline Selenese or PrimeFaces Inline Web Driver)
  • Press Execute - the methods will be generated to the Results tab.

Create a new class which extends PrimeFacesTest as per the following (for example TestAdminMenuExternalTabletAsAdmininstrator.java):

package sail.admin;

import org.junit.After;
import org.junit.Before;
import org.openqa.selenium.By;

import util.sail.BrowserConfiguration;
import util.sail.Devices;
import util.sail.PrimeFacesTest;

public class TestAdminMenuExternalTabletAsAdmininstrator extends PrimeFacesTest {

	private String pathToChromeDriver = "/Users/myUser/chromedriver.exe";
	
	@Before
	public void setup() throws Exception {
		setupChrome(new BrowserConfiguration().baseUrl("http://localhost:8080/myApplication/").pathToDriver(pathToChromeDriver).userAgentString(Devices.ipad.userAgentString));
	}
	
	@After
	public void teardown() {
		tearDownBrowser();
	}

	protected void login(String customer, String username, String password) throws Exception {

		driver.get(baseUrl);

		driver.findElement(By.name("customer")).clear();
		driver.findElement(By.name("customer")).sendKeys(customer);

		driver.findElement(By.name("user")).clear();
		driver.findElement(By.name("user")).sendKeys(username);

		driver.findElement(By.name("password")).clear();
		driver.findElement(By.name("password")).sendKeys(password);

		driver.findElement(By.cssSelector("input[type=\"submit\"]")).click();
	}
	
}

Note that because the login page is not generated by Skyve, it is not included in the SAIL generated methods.

Add your generated methods to this class, changing private methods to protected.

Create a new Test class which extends your class above (in this example TestAdminMenuExternalTabletAsAdmininstrator.java) and calls the desired methods, as per the following example:

package sail.admin;

import org.junit.Test;

public class TestAdminModule extends TestAdminMenuExternalTabletAsAdmininstrator {

	@Test
	public void test() throws Exception {
		
		String clearTextPassword = "Password01";
		
		login("demo", "admin", clearTextPassword);
		
		testMenuPassword();
		
	}
}

When running the test, use a JUnit run configuration, ensuring that the test\modules\test and lib jars are on the classpath.

Note on testing the admin module

A number of features in the admin module can have major effects on your data store and so testing the admin module should be performed with care. For example, the admin module gives a suitably credentialled user the ability to backup, restore, or truncate all database data, change passwords, delete users etc.

Automated User Interface testing impacts the persisted data and content of your application, and does not use an in-memory database (like automated unit testing).

The admin module distributed as part of the Skyve framework is tested as part of the Skyve release process, however, as developers can customise this module, they may need to test it, and Developers must consider carefully the implications of the UI tests they create.

⬆ back to top


Next Backup and restore
Previous Automated Unit Testing