2018-03-17 18:26:18 +05:30
|
|
|
# Frontend testing standards and style guidelines
|
|
|
|
|
|
|
|
There are two types of test suites you'll encounter while developing frontend code
|
|
|
|
at GitLab. We use Karma and Jasmine for JavaScript unit and integration testing,
|
|
|
|
and RSpec feature tests with Capybara for e2e (end-to-end) integration testing.
|
|
|
|
|
|
|
|
Unit and feature tests need to be written for all new features.
|
|
|
|
Most of the time, you should use [RSpec] for your feature tests.
|
|
|
|
|
|
|
|
Regression tests should be written for bug fixes to prevent them from recurring
|
|
|
|
in the future.
|
|
|
|
|
|
|
|
See the [Testing Standards and Style Guidelines](index.md) page for more
|
|
|
|
information on general testing practices at GitLab.
|
|
|
|
|
|
|
|
## Karma test suite
|
|
|
|
|
|
|
|
GitLab uses the [Karma][karma] test runner with [Jasmine] as its test
|
|
|
|
framework for our JavaScript unit and integration tests. For integration tests,
|
|
|
|
we generate HTML files using RSpec (see `spec/javascripts/fixtures/*.rb` for examples).
|
|
|
|
Some fixtures are still HAML templates that are translated to HTML files using the same mechanism (see `static_fixtures.rb`).
|
|
|
|
Adding these static fixtures should be avoided as they are harder to keep up to date with real views.
|
|
|
|
The existing static fixtures will be migrated over time.
|
|
|
|
Please see [gitlab-org/gitlab-ce#24753](https://gitlab.com/gitlab-org/gitlab-ce/issues/24753) to track our progress.
|
|
|
|
Fixtures are served during testing by the [jasmine-jquery][jasmine-jquery] plugin.
|
|
|
|
|
|
|
|
JavaScript tests live in `spec/javascripts/`, matching the folder structure
|
|
|
|
of `app/assets/javascripts/`: `app/assets/javascripts/behaviors/autosize.js`
|
|
|
|
has a corresponding `spec/javascripts/behaviors/autosize_spec.js` file.
|
|
|
|
|
|
|
|
Keep in mind that in a CI environment, these tests are run in a headless
|
|
|
|
browser and you will not have access to certain APIs, such as
|
|
|
|
[`Notification`](https://developer.mozilla.org/en-US/docs/Web/API/notification),
|
|
|
|
which will have to be stubbed.
|
|
|
|
|
|
|
|
### Best practices
|
|
|
|
|
|
|
|
#### Naming unit tests
|
|
|
|
|
|
|
|
When writing describe test blocks to test specific functions/methods,
|
|
|
|
please use the method name as the describe block name.
|
|
|
|
|
|
|
|
```javascript
|
|
|
|
// Good
|
|
|
|
describe('methodName', () => {
|
|
|
|
it('passes', () => {
|
|
|
|
expect(true).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// Bad
|
|
|
|
describe('#methodName', () => {
|
|
|
|
it('passes', () => {
|
|
|
|
expect(true).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// Bad
|
|
|
|
describe('.methodName', () => {
|
|
|
|
it('passes', () => {
|
|
|
|
expect(true).toEqual(true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
```
|
2018-10-15 14:42:47 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
#### Testing promises
|
|
|
|
|
|
|
|
When testing Promises you should always make sure that the test is asynchronous and rejections are handled.
|
|
|
|
Your Promise chain should therefore end with a call of the `done` callback and `done.fail` in case an error occurred.
|
|
|
|
|
|
|
|
```javascript
|
|
|
|
// Good
|
2018-10-15 14:42:47 +05:30
|
|
|
it('tests a promise', done => {
|
2018-03-17 18:26:18 +05:30
|
|
|
promise
|
2018-10-15 14:42:47 +05:30
|
|
|
.then(data => {
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(data).toBe(asExpected);
|
|
|
|
})
|
|
|
|
.then(done)
|
|
|
|
.catch(done.fail);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Good
|
2018-10-15 14:42:47 +05:30
|
|
|
it('tests a promise rejection', done => {
|
2018-03-17 18:26:18 +05:30
|
|
|
promise
|
|
|
|
.then(done.fail)
|
2018-10-15 14:42:47 +05:30
|
|
|
.catch(error => {
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(error).toBe(expectedError);
|
|
|
|
})
|
|
|
|
.then(done)
|
|
|
|
.catch(done.fail);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Bad (missing done callback)
|
|
|
|
it('tests a promise', () => {
|
2018-10-15 14:42:47 +05:30
|
|
|
promise.then(data => {
|
|
|
|
expect(data).toBe(asExpected);
|
|
|
|
});
|
2018-03-17 18:26:18 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
// Bad (missing catch)
|
2018-10-15 14:42:47 +05:30
|
|
|
it('tests a promise', done => {
|
2018-03-17 18:26:18 +05:30
|
|
|
promise
|
2018-10-15 14:42:47 +05:30
|
|
|
.then(data => {
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(data).toBe(asExpected);
|
|
|
|
})
|
2018-10-15 14:42:47 +05:30
|
|
|
.then(done);
|
2018-03-17 18:26:18 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
// Bad (use done.fail in asynchronous tests)
|
2018-10-15 14:42:47 +05:30
|
|
|
it('tests a promise', done => {
|
2018-03-17 18:26:18 +05:30
|
|
|
promise
|
2018-10-15 14:42:47 +05:30
|
|
|
.then(data => {
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(data).toBe(asExpected);
|
|
|
|
})
|
|
|
|
.then(done)
|
2018-10-15 14:42:47 +05:30
|
|
|
.catch(fail);
|
2018-03-17 18:26:18 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
// Bad (missing catch)
|
2018-10-15 14:42:47 +05:30
|
|
|
it('tests a promise rejection', done => {
|
2018-03-17 18:26:18 +05:30
|
|
|
promise
|
2018-10-15 14:42:47 +05:30
|
|
|
.catch(error => {
|
2018-03-17 18:26:18 +05:30
|
|
|
expect(error).toBe(expectedError);
|
|
|
|
})
|
2018-10-15 14:42:47 +05:30
|
|
|
.then(done);
|
2018-03-17 18:26:18 +05:30
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
#### Stubbing and Mocking
|
|
|
|
|
|
|
|
Jasmine provides useful helpers `spyOn`, `spyOnProperty`, `jasmine.createSpy`,
|
|
|
|
and `jasmine.createSpyObject` to facilitate replacing methods with dummy
|
|
|
|
placeholders, and recalling when they are called and the arguments that are
|
|
|
|
passed to them. These tools should be used liberally, to test for expected
|
|
|
|
behavior, to mock responses, and to block unwanted side effects (such as a
|
|
|
|
method that would generate a network request or alter `window.location`). The
|
|
|
|
documentation for these methods can be found in the [jasmine introduction page](https://jasmine.github.io/2.0/introduction.html#section-Spies).
|
|
|
|
|
|
|
|
Sometimes you may need to spy on a method that is directly imported by another
|
|
|
|
module. GitLab has a custom `spyOnDependency` method which utilizes
|
|
|
|
[babel-plugin-rewire](https://github.com/speedskater/babel-plugin-rewire) to
|
|
|
|
achieve this. It can be used like so:
|
|
|
|
|
|
|
|
```javascript
|
|
|
|
// my_module.js
|
|
|
|
import { visitUrl } from '~/lib/utils/url_utility';
|
|
|
|
|
|
|
|
export default function doSomething() {
|
|
|
|
visitUrl('/foo/bar');
|
|
|
|
}
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
// my_module_spec.js
|
|
|
|
import doSomething from '~/my_module';
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
describe('my_module', () => {
|
|
|
|
it('does something', () => {
|
|
|
|
const visitUrl = spyOnDependency(doSomething, 'visitUrl');
|
|
|
|
|
|
|
|
doSomething();
|
|
|
|
expect(visitUrl).toHaveBeenCalledWith('/foo/bar');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
|
|
|
Unlike `spyOn`, `spyOnDependency` expects its first parameter to be the default
|
|
|
|
export of a module who's import you want to stub, rather than an object which
|
|
|
|
contains a method you wish to stub (if the module does not have a default
|
|
|
|
export, one is be generated by the babel plugin). The second parameter is the
|
|
|
|
name of the import you wish to change. The result of the function is a Spy
|
|
|
|
object which can be treated like any other jasmine spy object.
|
|
|
|
|
|
|
|
Further documentation on the babel rewire pluign API can be found on
|
|
|
|
[its repository Readme doc](https://github.com/speedskater/babel-plugin-rewire#babel-plugin-rewire).
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
### Vue.js unit tests
|
|
|
|
|
|
|
|
See this [section][vue-test].
|
|
|
|
|
|
|
|
### Running frontend tests
|
|
|
|
|
|
|
|
`rake karma` runs the frontend-only (JavaScript) tests.
|
|
|
|
It consists of two subtasks:
|
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
* `rake karma:fixtures` (re-)generates fixtures
|
|
|
|
* `rake karma:tests` actually executes the tests
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
As long as the fixtures don't change, `rake karma:tests` (or `yarn karma`)
|
|
|
|
is sufficient (and saves you some time).
|
|
|
|
|
|
|
|
### Live testing and focused testing
|
|
|
|
|
|
|
|
While developing locally, it may be helpful to keep karma running so that you
|
2018-10-15 14:42:47 +05:30
|
|
|
can get instant feedback on as you write tests and modify code. To do this
|
|
|
|
you can start karma with `yarn run karma-start`. It will compile the javascript
|
2018-03-17 18:26:18 +05:30
|
|
|
assets and run a server at `http://localhost:9876/` where it will automatically
|
2018-10-15 14:42:47 +05:30
|
|
|
run the tests on any browser which connects to it. You can enter that url on
|
2018-03-17 18:26:18 +05:30
|
|
|
multiple browsers at once to have it run the tests on each in parallel.
|
|
|
|
|
|
|
|
While karma is running, any changes you make will instantly trigger a recompile
|
|
|
|
and retest of the entire test suite, so you can see instantly if you've broken
|
2018-10-15 14:42:47 +05:30
|
|
|
a test with your changes. You can use [jasmine focused][jasmine-focus] or
|
2018-03-17 18:26:18 +05:30
|
|
|
excluded tests (with `fdescribe` or `xdescribe`) to get karma to run only the
|
|
|
|
tests you want while you're working on a specific feature, but make sure to
|
|
|
|
remove these directives when you commit your code.
|
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
It is also possible to only run karma on specific folders or files by filtering
|
|
|
|
the run tests via the argument `--filter-spec` or short `-f`:
|
|
|
|
|
|
|
|
```bash
|
|
|
|
# Run all files
|
|
|
|
yarn karma-start
|
|
|
|
# Run specific spec files
|
|
|
|
yarn karma-start --filter-spec profile/account/components/update_username_spec.js
|
|
|
|
# Run specific spec folder
|
|
|
|
yarn karma-start --filter-spec profile/account/components/
|
|
|
|
# Run all specs which path contain vue_shared or vie
|
|
|
|
yarn karma-start -f vue_shared -f vue_mr_widget
|
|
|
|
```
|
|
|
|
|
|
|
|
You can also use glob syntax to match files. Remember to put quotes around the
|
|
|
|
glob otherwise your shell may split it into multiple arguments:
|
|
|
|
|
|
|
|
```bash
|
|
|
|
# Run all specs named `file_spec` within the IDE subdirectory
|
|
|
|
yarn karma -f 'spec/javascripts/ide/**/file_spec.js'
|
|
|
|
```
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
## RSpec feature integration tests
|
|
|
|
|
|
|
|
Information on setting up and running RSpec integration tests with
|
|
|
|
[Capybara] can be found in the [Testing Best Practices](best_practices.md).
|
|
|
|
|
|
|
|
## Gotchas
|
|
|
|
|
|
|
|
### Errors due to use of unsupported JavaScript features
|
|
|
|
|
|
|
|
Similar errors will be thrown if you're using JavaScript features not yet
|
|
|
|
supported by the PhantomJS test runner which is used for both Karma and RSpec
|
2018-10-15 14:42:47 +05:30
|
|
|
tests. We polyfill some JavaScript objects for older browsers, but some
|
2018-03-17 18:26:18 +05:30
|
|
|
features are still unavailable:
|
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
* Array.from
|
|
|
|
* Array.first
|
|
|
|
* Async functions
|
|
|
|
* Generators
|
|
|
|
* Array destructuring
|
|
|
|
* For..Of
|
|
|
|
* Symbol/Symbol.iterator
|
|
|
|
* Spread
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-10-15 14:42:47 +05:30
|
|
|
Until these are polyfilled appropriately, they should not be used. Please
|
2018-03-17 18:26:18 +05:30
|
|
|
update this list with additional unsupported features.
|
|
|
|
|
|
|
|
### RSpec errors due to JavaScript
|
|
|
|
|
|
|
|
By default RSpec unit tests will not run JavaScript in the headless browser
|
|
|
|
and will simply rely on inspecting the HTML generated by rails.
|
|
|
|
|
|
|
|
If an integration test depends on JavaScript to run correctly, you need to make
|
|
|
|
sure the spec is configured to enable JavaScript when the tests are run. If you
|
|
|
|
don't do this you'll see vague error messages from the spec runner.
|
|
|
|
|
|
|
|
To enable a JavaScript driver in an `rspec` test, add `:js` to the
|
|
|
|
individual spec or the context block containing multiple specs that need
|
|
|
|
JavaScript enabled:
|
|
|
|
|
|
|
|
```ruby
|
|
|
|
# For one spec
|
|
|
|
it 'presents information about abuse report', :js do
|
|
|
|
# assertions...
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "Admin::AbuseReports", :js do
|
|
|
|
it 'presents information about abuse report' do
|
|
|
|
# assertions...
|
|
|
|
end
|
|
|
|
it 'shows buttons for adding to abuse report' do
|
|
|
|
# assertions...
|
|
|
|
end
|
|
|
|
end
|
|
|
|
```
|
|
|
|
|
|
|
|
### Spinach errors due to missing JavaScript
|
|
|
|
|
|
|
|
NOTE: **Note:** Since we are discouraging the use of Spinach when writing new
|
2018-10-15 14:42:47 +05:30
|
|
|
feature tests, you shouldn't ever need to use this. This information is kept
|
2018-03-17 18:26:18 +05:30
|
|
|
available for legacy purposes only.
|
|
|
|
|
|
|
|
In Spinach, the JavaScript driver is enabled differently. In the `*.feature`
|
|
|
|
file for the failing spec, add the `@javascript` flag above the Scenario:
|
|
|
|
|
|
|
|
```
|
|
|
|
@javascript
|
|
|
|
Scenario: Developer can approve merge request
|
|
|
|
Given I am a "Shop" developer
|
|
|
|
And I visit project "Shop" merge requests page
|
|
|
|
And merge request 'Bug NS-04' must be approved
|
|
|
|
And I click link "Bug NS-04"
|
|
|
|
When I click link "Approve"
|
|
|
|
Then I should see approved merge request "Bug NS-04"
|
|
|
|
```
|
|
|
|
|
|
|
|
[jasmine-focus]: https://jasmine.github.io/2.5/focused_specs.html
|
|
|
|
[jasmine-jquery]: https://github.com/velesin/jasmine-jquery
|
|
|
|
[karma]: http://karma-runner.github.io/
|
2018-10-15 14:42:47 +05:30
|
|
|
[vue-test]: https://docs.gitlab.com/ce/development/fe_guide/vue.html#testing-vue-components
|
|
|
|
[rspec]: https://github.com/rspec/rspec-rails#feature-specs
|
|
|
|
[capybara]: https://github.com/teamcapybara/capybara
|
|
|
|
[karma]: http://karma-runner.github.io/
|
|
|
|
[jasmine]: https://jasmine.github.io/
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
[Return to Testing documentation](index.md)
|