2016-02-06 02:27:06 -05:00
include ../_util-fns
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
:marked
2016-01-19 04:15:26 -05:00
In this chapter we'll setup the environment for testing our sample application and write a few easy Jasmine tests of the app's simplest parts.
2015-10-14 23:25:19 -04:00
We'll learn:
2016-03-24 15:30:26 -04:00
- to test one of our application files
2015-10-14 23:25:19 -04:00
- why we prefer our test files to be next to their corresponding source files
- to run tests with an `npm` command
2016-02-11 18:08:06 -05:00
- load the test file with SystemJS
2015-10-14 23:25:19 -04:00
.callout.is-helpful
header Prior Knowledge
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
The Unit Testing chapters build upon each other. We recommend reading them in order.
We're also assuming that you're already comfortable with basic Angular 2 concepts and the tools
we introduced in the [QuickStart](../quickstart.html) and
2015-10-15 02:04:58 -04:00
the [Tour of Heroes](../tutorial/) tutorial
2015-10-14 23:25:19 -04:00
such as <code>npm</code>, <code>gulp</code>, and <code>live-server</code>.
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## Create the test-runner HTML
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
Locate the folder that contains the application `index.html` for your testing copy of Tour of Heroes.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
Create a new, sibling HTML file, ** `unit-tests.html` ** and copy over the same basic material from the `unit-tests.html` in the [Jasmine 101](./jasmine-testing-101.html) chapter.
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/unit-tests-2.html', 'test-runner-base', 'unit-tests.html')
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
:marked
2016-01-19 04:15:26 -05:00
We're picking up right where we left off. All we've done is change the title.
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## Update `package.json` for testing
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
We must install the Jasmine package as well:
2015-10-09 13:33:12 -04:00
pre.prettyprint.lang-bash
2015-10-14 23:25:19 -04:00
code npm install jasmine-core --save-dev --save-exact
2015-10-09 13:33:12 -04:00
2015-10-10 06:29:34 -04:00
.alert.is-important Be sure to install <code>jasmine-core</code> , not <code>jasmine</code>!
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
:marked
2016-01-19 04:15:26 -05:00
Let's make one more change to the `package.json` script commands.
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
**Open the `package.json` ** and scroll to the `scripts` node and add in a new one:
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
code-example(format="").
"test": "live-server --open=unit-tests.html"
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
:marked
2015-10-14 23:25:19 -04:00
That command will launch `live-server` and open a browser to the `unit-tests.html` page we just wrote.
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## First app tests
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
We can start testing *some* of our app right away. For example, we can test the `Hero` interface:
2016-03-04 20:56:41 -05:00
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/app/hero.ts')
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
:marked
Let's add a couple of simple tests in a new file.
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/app/hero.spec.ts', 'base-hero-spec')
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
:marked
2016-01-19 04:15:26 -05:00
That's the basic Jasmine we learned back in "Jasmine 101".
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
Notice that we surrounded our tests with ** `describe('Hero')` **.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
**By convention, our test always begin with a `describe` that identifies the application part under test.**
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
The description should be sufficient to identify the tested application part and its source file. Almost any convention will do as long as you and your team follow it consistently and are never confused.
2016-03-17 10:07:54 -04:00
But we haven't saved this test yet.
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## Where do tests go?
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
Some people like to keep their tests in a `tests` folder parallel to the application source folder.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
We are not those people. We like our unit tests to be close to the source code that they test. We prefer this approach because
- The tests are easy to find
- We see at a glance if an application part lacks tests.
- Nearby tests can teach us about how the part works; they express the developers intention and reveal how the developer thinks the part should behave under a variety of circumstances.
- When we move the source (inevitable), we remember to move the test.
- When we rename the source file (inevitable), we remember to rename the test file.
2015-10-09 13:33:12 -04:00
2016-01-19 04:15:26 -05:00
We can't think of a downside. The server doesn't care where they are. They are easy to find and distinguish from application files when named conventionally.
2015-10-09 13:33:12 -04:00
2016-02-06 02:27:06 -05:00
.l-sub-section
:marked
You may put your tests elsewhere if you wish.
We're putting ours inside the app, next to the source files that they test.
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## First spec file
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
**Create** a new file, ** `hero.spec.ts` ** in `app` next to `hero.ts`.
2015-10-09 13:33:12 -04:00
2016-01-19 04:15:26 -05:00
Notice the ".spec" suffix in the test file's filename, appended to the name of the file holding the application part we're testing.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
.alert.is-important All of our unit test files follow this .spec naming pattern.
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
:marked
2016-03-17 10:07:54 -04:00
Save the tests we just made in `hero.spec.ts`:
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/app/hero.spec.ts', 'base-hero-spec')
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
:marked
2016-01-19 04:15:26 -05:00
### Import the part we're testing
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
We have an `import {Hero} from './hero' ` statement.
2015-10-09 13:33:12 -04:00
2016-03-24 15:30:26 -04:00
If we forgot this import, a TypeScript-aware editor would warn us, with a squiggly red underline, that it can't find the definition of the `Hero` interface.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
### Update unit-tests.html
2015-10-09 13:33:12 -04:00
2016-02-29 00:18:31 -05:00
Next we update the `unit-tests.html` with a reference to our new `hero.spec.ts` file. Delete the inline test code. The revised pertinent HTML looks like this:
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/unit-tests-2.html', 'load-hero-and-spec')(format=".")
2016-03-04 20:56:41 -05:00
:marked
2015-10-16 04:06:56 -04:00
### Run and Fail
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
Look over at the browser (live-server will have reloaded it). The browser displays
2015-10-09 13:33:12 -04:00
figure.image-display
2015-10-14 23:25:19 -04:00
img(src='/resources/images/devguide/first-app-tests/Jasmine-not-running-tests.png' style="width:400px;" alt="Jasmine not running any tests")
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
:marked
2016-01-19 04:15:26 -05:00
That's Jasmine saying "**things are _so_ bad that _I'm not running any tests_.**"
2015-10-09 13:33:12 -04:00
2016-01-19 04:15:26 -05:00
Open the browser's Developer Tools (F12, Ctrl-Shift-i). There's an error:
2015-11-10 13:31:46 -05:00
2015-10-16 04:06:56 -04:00
code-example(format="" language="html").
2016-03-17 10:07:54 -04:00
Uncaught ReferenceError: System is not defined
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2016-02-11 18:08:06 -05:00
## Load tests with SystemJS
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
The immediate cause of the error is the `export` statement in `hero.ts`.
That error was there all along.
2016-03-24 15:30:26 -04:00
It wasn't a problem until we tried to `import` the `Hero` interface in our tests.
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
Our test environment lacks support for module loading.
2016-03-17 10:07:54 -04:00
Apparently we can't simply load our application and test scripts like we do with 3rd party JavaScript libraries.
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
We are committed to module loading in our application.
2015-10-14 23:25:19 -04:00
Our app will call `import`. Our tests must do so too.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
We add module loading support in four steps:
2015-10-09 13:33:12 -04:00
2016-02-11 18:08:06 -05:00
1. add the *SystemJS* module management library
1. configure *SystemJS* to look for JavaScript files by default
2015-10-14 23:25:19 -04:00
1. import our test files
1. tell Jasmine to run the imported tests
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
These steps are all clearly visible, in exactly that order, in the following lines that
replace the `<body>` contents in `unit-tests.html`:
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
+makeExample('testing/ts/unit-tests-3.html', 'systemjs')(format=".")
2015-10-09 13:33:12 -04:00
2016-03-17 10:07:54 -04:00
:marked
2015-10-14 23:25:19 -04:00
Look in the browser window. Our tests pass once again.
2015-10-09 13:33:12 -04:00
figure.image-display
2015-10-14 23:25:19 -04:00
img(src='/resources/images/devguide/first-app-tests/test-passed-once-again.png' style="width:400px;" alt="Tests passed once again")
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2015-10-14 23:25:19 -04:00
## Observations
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
### System.config
2015-11-10 13:31:46 -05:00
System.js demands that we specify a default extension for the filenames that correspond to whatever it is asked to import.
2016-01-19 04:15:26 -05:00
Without that default, it would translate an import statement such as `import {Hero} from './hero'` to a request for the file named `hero`.
Not `hero.js`. Just plain `hero`. Our server error with "404 - not found" because it doesn't have a file of that name.
2015-10-09 13:33:12 -04:00
2016-02-11 18:08:06 -05:00
Once configured with a default extension of 'js', SystemJS requests `hero.js` which *does* exist and is promptly returned by our server.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
### Asynchronous System.import
2016-06-21 11:55:52 -04:00
The call to `System.import` shouldn't surprise us but its asynchronous nature might.
2015-11-10 13:31:46 -05:00
If we ponder this for a moment, we realize that it must be asynchronous because
2015-10-14 23:25:19 -04:00
System.js may have to fetch the corresponding JavaScript file from the server.
2015-11-10 13:31:46 -05:00
Accordingly, `System.import` returns a promise and we must wait for that promise to resolve.
2015-10-14 23:25:19 -04:00
Only then can Jasmine start evaluating the imported tests.
2015-10-09 13:33:12 -04:00
2015-10-14 23:25:19 -04:00
### window.onload
2016-01-19 04:15:26 -05:00
Jasmine doesn't have a `start` method. It wires its own start to the browser window's `load` event.
That makes sense if we're loading our tests with script tags.
2015-10-14 23:25:19 -04:00
The browser raises the `load` event when it finishes loading all scripts.
2015-10-09 13:33:12 -04:00
2016-01-19 04:15:26 -05:00
But we're not loading test scripts inline anymore.
2016-02-11 18:08:06 -05:00
We're using the SystemJS module loader and it won't be done until long after the browser raised the `load` event.
2015-10-14 23:25:19 -04:00
Meanwhile, Jasmine started and ran to completion … with no tests to evaluate … before the import completed.
2015-10-09 13:33:12 -04:00
2015-11-10 13:31:46 -05:00
So we must wait until the import completes and only then call the window `onLoad` handler.
2015-10-14 23:25:19 -04:00
Jasmine re-starts, this time with our imported test queued up.
2015-10-09 13:33:12 -04:00
2015-10-16 04:06:56 -04:00
.l-main-section
2015-11-10 13:31:46 -05:00
:marked
2016-01-19 04:15:26 -05:00
## What's Next?
2015-11-10 13:31:46 -05:00
We are able to test a part of our application with simple Jasmine tests.
2016-03-24 15:30:26 -04:00
The part was a stand-alone interface that made no mention or use of Angular.
2015-10-09 13:33:12 -04:00
2016-02-06 02:27:06 -05:00
That's not rare but it's not typical either.
Most of our application parts make some use of the Angular framework.
Let's test a *pipe* class that does rely on Angular.