diff --git a/pages/en/lb4/Testing-your-application.md b/pages/en/lb4/Testing-your-application.md index defb28176..679627ccc 100644 --- a/pages/en/lb4/Testing-your-application.md +++ b/pages/en/lb4/Testing-your-application.md @@ -13,29 +13,40 @@ summary: A thorough automated test suite is important because it: - Ensures your application works as expected. - Prevents regressions when new features are added and bugs are fixed. -- Helps new and existing developers understand different parts of the codebase (knowledge sharing). +- Helps new and existing developers understand different parts of the codebase +(knowledge sharing). - Speeds up development over the long run (the code writes itself!) ### Types of tests -We encourage writing tests from a few perspectives, mainly [black-box testing](https://en.wikipedia.org/wiki/Black-box_testing) (acceptance) and [white-box testing](https://en.wikipedia.org/wiki/White-box_testing) (integration and unit). Tests are usually written using typical patterns such as [`arrange/act/assert`](https://msdn.microsoft.com/en-us/library/hh694602.aspx#Anchor_3) or [`given/when/then`](https://martinfowler.com/bliki/GivenWhenThen.html). While both styles work well, just pick one that you're comfortable with and start writing tests! +We encourage writing tests from a few perspectives, mainly [black-box testing](https://en.wikipedia.org/wiki/Black-box_testing) +(acceptance) and [white-box testing](https://en.wikipedia.org/wiki/White-box_testing) +(integration and unit). Tests are usually written using typical patterns such as +[`arrange/act/assert`](https://msdn.microsoft.com/en-us/library/hh694602.aspx#Anchor_3) +or [`given/when/then`](https://martinfowler.com/bliki/GivenWhenThen.html). +While both styles work well, just pick one that you're comfortable with and +start writing tests! -For an introduction to automated testing, see [Define your testing strategy](Defining-your-testing-strategy.html); for a step-by-step tutorial, see [Incrementally implement features](Implementing-features.html). +For an introduction to automated testing, see [Define your testing strategy](Defining-your-testing-strategy.html); +for a step-by-step tutorial, see [Incrementally implement features](Implementing-features.html). {% include important.html content=" -A great test suite requires you to think smaller and favor fast and focused unit tests over slow end-to-end tests. + A great test suite requires you to think smaller and favor fast and focused + unit tests over slow end-to-end tests. " %} This article is a reference guide for common types of tests and test helpers. ## Project setup -An automated test suite requires a test runner to execute all the tests and produce a summary report. We use and recommend [Mocha](https://mochajs.org). +An automated test suite requires a test runner to execute all the tests and +produce a summary report. We use and recommend [Mocha](https://mochajs.org). In addition to a test runner, the test suites generally requires: - An assertion library (we recommend [Should.js](https://shouldjs.github.io)). -- A Library for making HTTP calls and verifying their results (we recommend [supertest](https://github.com/visionmedia/supertest)). +- A Library for making HTTP calls and verifying their results (we recommend +[supertest](https://github.com/visionmedia/supertest)). - A library for creating test doubles (we recommend [Sinon.JS](http://sinonjs.org/)). The [@loopback/testlab](https://www.npmjs.com/package/@loopback/testlab) module @@ -76,15 +87,27 @@ Your `package.json` should then look something like this: ## Data handling -Tests accessing a real database often require existing data. For example, a method listing all products needs some products in the database; a method to create a new product instance must determine which properties are required and any restrictions on their values. There are various approaches to address this issue. Many of them unfortunately make the test suite difficult to understand, difficult to maintain, and prone to test failures unrelated to the changes made. +Tests accessing a real database often require existing data. For example, +a method listing all products needs some products in the database; a method +to create a new product instance must determine which properties are required +and any restrictions on their values. There are various approaches to address +this issue. Many of them unfortunately make the test suite difficult +to understand, difficult to maintain, and prone to test failures unrelated +to the changes made. Based on our experience, we recommend the following approach. ### Clean the database before each test -Always start with a clean database before each test. This may seem counter-intuitive: why not reset the database after the test has finished? When a test fails and the database is cleaned after the test has finished, then it's difficult to observe what was stored in the database and why the test failed. When the database is cleaned in the beginning, then any failing test will leave the database in the state that caused the test to fail. +Always start with a clean database before each test. This may seem +counter-intuitive: why not reset the database after the test has finished? +When a test fails and the database is cleaned after the test has finished, +then it's difficult to observe what was stored in the database and why the test +failed. When the database is cleaned in the beginning, then any failing test +will leave the database in the state that caused the test to fail. -To clean the database before each test, set up a `beforeEach` hook to call a helper method; for example: +To clean the database before each test, set up a `beforeEach` hook to call +a helper method; for example: {% include code-caption.html content="test/helpers/database.helpers.ts" %} ```ts @@ -102,15 +125,29 @@ describe('ProductController (integration)', () => { ### Use test data builders -To avoid duplicating code for creating model data with all required properties filled in, use shared [test data builders](http://www.natpryce.com/articles/000714.html) instead. This enables tests to provide a small subset of properties that are strictly required by the tested scenario, which is important because it makes tests: +To avoid duplicating code for creating model data with all required properties +filled in, use shared [test data builders](http://www.natpryce.com/articles/000714.html) +instead. This enables tests to provide a small subset of properties that are +strictly required by the tested scenario, which is important because +it makes tests: -- Easier to understand, since it's immediately clear what model properties are relevant to the test. If the test were setting all required properties, it would be difficult to tell whether some of those properties are actually relevant to the tested scenario. +- Easier to understand, since it's immediately clear what model properties are +relevant to the test. If the test were setting all required properties, +it would be difficult to tell whether some of those properties are actually +relevant to the tested scenario. -- Easier to maintain. As your data model evolves, you eventually need to add more required properties. If the tests were building model instance data manually, you would have to fix all tests to set the new required property. With a shared helper, there is only a single place where to add a value for the new required property. +- Easier to maintain. As your data model evolves, you eventually need to add +more required properties. If the tests were building model instance data +manually, you would have to fix all tests to set the new required property. +With a shared helper, there is only a single place where to add a value for the +new required property. -See [@loopback/openapi-spec-builder](https://www.npmjs.com/package/@loopback/openapi-spec-builder) for an example of how to apply this design pattern for building OpenAPI Spec documents. +See [@loopback/openapi-spec-builder](https://www.npmjs.com/package/@loopback/openapi-spec-builder) +for an example of how to apply this design pattern for building OpenAPI Spec +documents. -In practice, a rich method-based API is overkill and a simple function that adds missing required properties is sufficient. +In practice, a rich method-based API is overkill and a simple function that adds +missing required properties is sufficient. {% include code-caption.html content="test/helpers/database.helpers.ts" %} ```ts @@ -132,18 +169,43 @@ export async function givenProduct(data?: Partial) { ### Avoid sharing the same data for multiple tests -It's tempting to define a small set of data that's shared by all tests. For example, in an e-commerce application, you might pre-populate the database with few categories, some products, an admin user and a customer. Such approach has several downsides: - -- When trying to understand any individual test, it's difficult to tell what part of the pre-populated data is essential for the test and what's irrelevant. For example, in a test checking the method counting the number of products in a category using a pre-populated category "Stationery", is it important that "Stationery" contains nested sub-categories or is that fact irrelevant? If it's irrelevant, then what are the other tests that depend on it? - -- As the application grows and new features are added, it's easier to add more properties to existing model instances rather than create new instances using only properties required by the new features. For example, when adding a category image, it's easier to add image to an existing category "Stationery" and perhaps keep another category "Groceries" without any image, rather than create two new categories "CategoryWithAnImage" and "CategoryMissingImage". This further amplifies the previous problem, because it's not clear that "Groceries" is the category that should be used by tests requiring a category with no image - the category name does not provide any hints on that. - -- As the shared dataset grows (together with the application), the time required to bring the database into initial state grows too. Instead of running a few "DELETE ALL" queries before each test (which is relatively fast), you can end up with running tens to hundreds different commands creating different model instances, triggering slow index rebuilds along the way, and considerably slowing the test suite. - -Use the test data builders described in the previous section to populate your database with the data specific to your test only. - - - + -Write higher-level helpers to share the code for re-creating common scenarios. For example, if your application has two kinds of users (admins and customers), then you may write the following helpers to simplify writing acceptance tests checking access control: +Write higher-level helpers to share the code for re-creating common scenarios. +For example, if your application has two kinds of users (admins and customers), +then you may write the following helpers to simplify writing acceptance tests +checking access control: ```ts async function givenAdminAndCustomer() { @@ -188,33 +253,70 @@ async function givenAdminAndCustomer() { ## Unit testing -Unit tests are considered "white-box" tests because they use an "inside-out" approach where the tests know about the internals and controls all the variables of the system being tested. Individual units are tested in isolation, their dependencies are replaced with [Test doubles](https://en.wikipedia.org/wiki/Test_double). +Unit tests are considered "white-box" tests because they use an "inside-out" +approach where the tests know about the internals and controls all the variables +of the system being tested. Individual units are tested in isolation, +their dependencies are replaced with [Test doubles](https://en.wikipedia.org/wiki/Test_double). ### Use test doubles -Test doubles are functions or objects that look and behave like the real variants used in production, but are actually simplified versions giving the test more control of the behavior. For example, reproducing the situation where reading from a file failed because of a hard-drive error is pretty much impossible, unless we are using a test double that's simulating file-system API and giving us control of how what each call returns. +Test doubles are functions or objects that look and behave like the real +variants used in production, but are actually simplified versions giving the +test more control of the behavior. For example, reproducing the situation where +reading from a file failed because of a hard-drive error is pretty much +impossible, unless we are using a test double that's simulating file-system API +and giving us control of how what each call returns. -[Sinon.JS](http://sinonjs.org/) has become the de-facto standard for test doubles in Node.js and JavaScript/TypeScript in general. The `@loopback/testlab` package comes with Sinon preconfigured with TypeScript type definitions and integrated with Should.js assertions. +[Sinon.JS](http://sinonjs.org/) has become the de-facto standard for +test doubles in Node.js and JavaScript/TypeScript in general. +The `@loopback/testlab` package comes with Sinon preconfigured with TypeScript +type definitions and integrated with Should.js assertions. There are three kinds of test doubles provided by Sinon.JS: -- [Test spies](http://sinonjs.org/releases/v4.0.1/spies/) are functions that record arguments, the return value, the value of `this`, and exceptions thrown (if any) for all its calls. There are two types of spies: Some are anonymous functions, while others wrap methods that already exist in the system under test. +- [Test spies](http://sinonjs.org/releases/v4.0.1/spies/) are functions that +record arguments, the return value, the value of `this`, and exceptions thrown +(if any) for all its calls. There are two types of spies: Some are +anonymous functions, while others wrap methods that already exist in the system +under test. -- [Test stubs](http://sinonjs.org/releases/v4.0.1/stubs/) are functions (spies) with pre-programmed behavior. As spies, stubs can be either anonymous, or wrap existing functions. When wrapping an existing function with a stub, the original function is not called. +- [Test stubs](http://sinonjs.org/releases/v4.0.1/stubs/) are functions (spies) +with pre-programmed behavior. As spies, stubs can be either anonymous, or wrap +existing functions. When wrapping an existing function with a stub, the original +function is not called. -- [Test mocks](http://sinonjs.org/releases/v4.0.1/mocks/) (and mock expectations) are fake methods (like spies) with pre-programmed behavior (like stubs) as well as pre-programmed expectations. A mock will fail your test if it is not used as expected. +- [Test mocks](http://sinonjs.org/releases/v4.0.1/mocks/) +(and mock expectations) are fake methods (like spies) with pre-programmed +behavior (like stubs) as well as pre-programmed expectations. A mock will fail +your test if it is not used as expected. {% include note.html content=" -We recommend against using test mocks. With test mocks, the expectations must be defined before the tested scenario is executed, which breaks the recommended test layout 'arrange-act-assert' (or 'given-when-then') and produces code that's difficult to comprehend. + We recommend against using test mocks. With test mocks, the expectations must + be defined before the tested scenario is executed, which breaks the + recommended test layout 'arrange-act-assert' (or 'given-when-then') and + produces code that's difficult to comprehend. " %} #### Create a stub Repository -When writing an application accessing data in a database, best practice is to use [repositories](Repositories.html) to encapsulate all data-access/persistence-related code and let other parts of the application (typically [controllers](Controllers.html)) to depend on these repositories for data access. To test Repository dependents (for example, Controllers) in isolation, we need to provide a test double, usually as a test stub. - -In traditional object-oriented languages like Java or C#, to enable unit tests to provide a custom implementation of the repository API, the controller needs to depend on an interface describing the API, and the repository implementation needs to implement this interface. The situation is easier in JavaScript and TypeScript. Thanks to the dynamic nature of the language, it’s possible to mock/stub entire classes. - -Creating a test double for a repository class is very easy using the Sinon.JS utility function `createStubInstance`. It's important to create a new stub instance for each unit test in order to prevent unintended re-use of pre-programmed behavior between (unrelated) tests. +When writing an application accessing data in a database, best practice is +to use [repositories](Repositories.html) to encapsulate all +data-access/persistence-related code and let other parts of the application +(typically [controllers](Controllers.html)) to depend on these repositories for +data access. To test Repository dependents (for example, Controllers) in +isolation, we need to provide a test double, usually as a test stub. + +In traditional object-oriented languages like Java or C#, to enable unit tests +to provide a custom implementation of the repository API, the controller needs +to depend on an interface describing the API, and the repository implementation +needs to implement this interface. The situation is easier in JavaScript and +TypeScript. Thanks to the dynamic nature of the language, it’s possible to +mock/stub entire classes. + +Creating a test double for a repository class is very easy using the Sinon.JS +utility function `createStubInstance`. It's important to create a new stub +instance for each unit test in order to prevent unintended re-use of +pre-programmed behavior between (unrelated) tests. ```ts describe('ProductController', () => { @@ -229,9 +331,12 @@ describe('ProductController', () => { }); ``` -In your unit tests, you will usually want to program the behavior of stubbed methods (what they should return) and then verify that the Controller (unit under test) called the right method with the correct arguments. +In your unit tests, you will usually want to program the behavior of stubbed +methods (what they should return) and then verify that the Controller +(unit under test) called the right method with the correct arguments. -Configure stub's behavior at the beginning of your unit test (in the "arrange" or "given" section): +Configure stub's behavior at the beginning of your unit test +(in the "arrange" or "given" section): ```ts // repository.find() will return a promise that @@ -240,7 +345,8 @@ const findStub = repository.find as sinon.SinonStub; findStub.resolves([{id: 1, name: 'Pen'}]); ``` -Verify how was the stubbed method executed at the end of your unit test (in the "assert" or "then" section): +Verify how was the stubbed method executed at the end of your unit test +(in the "assert" or "then" section): ```ts // expect that repository.find() was called with the first @@ -248,17 +354,26 @@ Verify how was the stubbed method executed at the end of your unit test (in the sinon.assert.calledWithMatch({where: {id: 1}}); ``` -See [Unit test your controllers](#unit-test-your-controllers) for a full example. +See [Unit test your controllers](#unit-test-your-controllers) for a +full example. #### Create a stub Service {% include content/tbd.html %} -To be done. The initial beta release does not include Services as a first-class feature. +To be done. The initial beta release does not include Services as a first-class +feature. ### Unit test your Controllers -Unit tests should apply to the smallest piece of code possible to ensure other variables and state changes do not pollute the result. A typical unit test creates a controller instance with dependencies replaced by test doubles and directly calls the tested method. The example below gives the controller a stub implementation of its repository dependency, and then ensure the controller called repository's `find()` method with a correct query and returned back the query results. See [Create a stub repository](#create-a-stub-repository) for a detailed explanation. +Unit tests should apply to the smallest piece of code possible to ensure other +variables and state changes do not pollute the result. A typical unit test +creates a controller instance with dependencies replaced by test doubles and +directly calls the tested method. The example below gives the controller a stub +implementation of its repository dependency, and then ensure the controller +called repository's `find()` method with a correct query and returned back the +query results. See [Create a stub repository](#create-a-stub-repository) +for a detailed explanation. {% include code-caption.html content="test/unit/controllers/product.controller.test.ts" %} ```ts @@ -291,11 +406,17 @@ describe('ProductController (unit)', () => { ### Unit test your models and repositories -In a typical LoopBack application, models and repositories rely on behavior provided by the framework (`@loopback/repository` package) and there is no need to test LoopBack's built-in functionality. However, any additional application-specific API does need new unit tests. +In a typical LoopBack application, models and repositories rely on behavior +provided by the framework (`@loopback/repository` package) and there is no need +to test LoopBack's built-in functionality. However, any additional +application-specific API does need new unit tests. -For example, if the `Person` Model has properties `firstname`, `middlename` and `surname` and provides a function to obtain the full name, then you should write unit tests to verify the implementation of this additional method. +For example, if the `Person` Model has properties `firstname`, `middlename` and +`surname` and provides a function to obtain the full name, then you should write +unit tests to verify the implementation of this additional method. -Remember to use [Test data builders](#use-test-data-builders) whenever you need valid data to create a new model instance. +Remember to use [Test data builders](#use-test-data-builders) whenever you need +valid data to create a new model instance. {% include code-caption.html content="test/unit/models/person.model.test.ts" %} @@ -335,42 +456,64 @@ describe('Person (unit)', () => { }); ``` -Writing a unit test for a custom repository methods is not as straightforward because `CrudRepository` is based on legacy [loopback-datasource-juggler](https://github.com/strongloop/loopback-datasource-juggler) which was not designed with dependency injection in mind. Instead, use integration tests to verify the implementation of custom repository methods; see [Test your repositories against a real database](#test-your-repositories-against-a-real-database) in [Integration Testing](#integration-testing). +Writing a unit test for a custom repository methods is not as straightforward +because `CrudRepository` is based on legacy [loopback-datasource-juggler](https://github.com/strongloop/loopback-datasource-juggler) +which was not designed with dependency injection in mind. Instead, use +integration tests to verify the implementation of custom repository methods; +see [Test your repositories against a real database](#test-your-repositories-against-a-real-database) +in [Integration Testing](#integration-testing). ### Unit test your Sequence -While it's possible to test a custom Sequence class in isolation, it's better to rely on acceptance-level tests in this exceptional case. The reason is that a custom Sequence class typically has many dependencies (which can make test setup too long and complex), and at the same time it provides very little functionality on top of the injected sequence actions. Bugs are much more likely to be caused by the way the real sequence action implementations interact together (which is not covered by unit tests), instead of the Sequence code itself (which is the only thing covered). +While it's possible to test a custom Sequence class in isolation, it's better +to rely on acceptance-level tests in this exceptional case. The reason is that +a custom Sequence class typically has many dependencies (which can make test +setup too long and complex), and at the same time it provides very little +functionality on top of the injected sequence actions. Bugs are much more likely +to be caused by the way the real sequence action implementations interact +together (which is not covered by unit tests), instead of the Sequence code +itself (which is the only thing covered). -See [Test Sequence customizations](#test-sequence-customizations) in [Acceptance Testing](#acceptance-end-to-end-testing). +See [Test Sequence customizations](#test-sequence-customizations) in +[Acceptance Testing](#acceptance-end-to-end-testing). ### Unit test your Services {% include content/tbd.html %} -To be done. The initial beta release does not include Services as a first-class feature. +To be done. The initial beta release does not include Services as a +first-class feature. See the following related GitHub issues: - - Define services to represent interactions with REST APIs, SOAP Web Services, gRPC services, and more: [#522](https://github.com/strongloop/loopback-next/issues/522) + - Define services to represent interactions with REST APIs, SOAP Web Services, + gRPC services, and more: [#522](https://github.com/strongloop/loopback-next/issues/522) - Guide: Services [#451](https://github.com/strongloop/loopback-next/issues/451) ## Integration testing -Integration tests are considered "white-box" tests because they use an "inside-out" approach that tests how multiple units work together or with external services. You can use test doubles to isolate tested units from external variables/state that are not part of the tested scenario. +Integration tests are considered "white-box" tests because they use an +"inside-out" approach that tests how multiple units work together or with +external services. You can use test doubles to isolate tested units from +external variables/state that are not part of the tested scenario. ### Test your repositories against a real database There are two common reasons for adding repository tests: - - Your models are using advanced configuration, for example, custom column mappings, and you want to verify this configuration is correctly picked up by the framework. + - Your models are using advanced configuration, for example, custom column + mappings, and you want to verify this configuration is correctly picked up by + the framework. - Your repositories have additional methods. -Integration tests are one of the places to put the best practices in [Data handling](#data-handling) to work: +Integration tests are one of the places to put the best practices in +[Data handling](#data-handling) to work: - Clean the database before each test - Use test data builders - Avoid sharing the same data for multiple tests -Here is an example showing how to write an integration test for a custom repository method `findByName`: +Here is an example showing how to write an integration test for a custom +repository method `findByName`: {% include code-caption.html content= "test/integration/repositories/category.repository.test.ts" %} ```ts @@ -400,7 +543,11 @@ describe('CategoryRepository (integration)', () => { ### Test controllers and repositories together -Integration tests running controllers with real repositories are important to verify that the controllers use the repository API correctly, and the commands and queries produce expected results when executed on a real database. These tests are similar to repository tests: we are just adding controllers as another ingredient. +Integration tests running controllers with real repositories are important to +verify that the controllers use the repository API correctly, and the commands +and queries produce expected results when executed on a real database. +These tests are similar to repository tests: we are just adding controllers as +another ingredient. ```ts import {expect} from '@loopback/testlab'; @@ -429,14 +576,22 @@ describe('ProductController (integration)', () => { {% include content/tbd.html %} -To be done. The initial beta release does not include Services as a first-class feature. +To be done. The initial beta release does not include Services as a first-class +feature. ## Acceptance (end-to-end) testing -Automated acceptance (end-to-end) tests are considered "black-box" tests because they use an "outside-in" approach that is not concerned about the internals of the system, just simply do the same actions (send the same HTTP requests) as the clients and consumers of your API will do, and verify the results returned by the system under test are matching the expectations. +Automated acceptance (end-to-end) tests are considered "black-box" tests because +they use an "outside-in" approach that is not concerned about the internals of +the system, just simply do the same actions (send the same HTTP requests) as the +clients and consumers of your API will do, and verify the results returned by +the system under test are matching the expectations. -Typically, acceptance tests start the application, make HTTP requests to the server, and verify the returned response. LoopBack uses [supertest](https://github.com/visionmedia/supertest) to make the test code that executes HTTP requests and verifies responses easier to write and read. -Remember to follow the best practices from [Data handling](#data-handling) when setting up your database for tests: +Typically, acceptance tests start the application, make HTTP requests to the +server, and verify the returned response. LoopBack uses [supertest](https://github.com/visionmedia/supertest) +to make the test code that executes HTTP requests and verifies responses easier +to write and read. Remember to follow the best practices from [Data handling](#data-handling) +when setting up your database for tests: - Clean the database before each test - Use test data builders @@ -444,8 +599,14 @@ Remember to follow the best practices from [Data handling](#data-handling) when ### Validate your OpenAPI specification -The OpenAPI specification is a cornerstone of applications that provide REST APIs. -It enables API consumers to leverage a whole ecosystem of related tooling. To make the spec useful, you must ensure it's a valid OpenAPI Spec document, ideally in an automated way that's an integral part of regular CI builds. LoopBack's [testlab](https://www.npmjs.com/package/@loopback/testlab) module provides a helper method `validateApiSpec` that builds on top of the popular [swagger-parser](https://www.npmjs.com/package/swagger-parser) package. +The OpenAPI specification is a cornerstone of applications that provide +REST APIs. +It enables API consumers to leverage a whole ecosystem of related tooling. +To make the spec useful, you must ensure it's a valid OpenAPI Spec document, +ideally in an automated way that's an integral part of regular CI builds. +LoopBack's [testlab](https://www.npmjs.com/package/@loopback/testlab) module +provides a helper method `validateApiSpec` that builds on top of the popular +[swagger-parser](https://www.npmjs.com/package/swagger-parser) package. Example usage: @@ -475,9 +636,17 @@ describe('API specification', () => { after our MVP release. " %} -The formal validity of your application's spec does not guarantee that your implementation is actually matching the specified behavior. To keep your spec in sync with your implementation, you should use an automated tool like [Dredd](https://www.npmjs.com/package/dredd) to run a set of smoke tests to verify conformance of your app with the spec. +The formal validity of your application's spec does not guarantee that your +implementation is actually matching the specified behavior. To keep your spec +in sync with your implementation, you should use an automated tool like [Dredd](https://www.npmjs.com/package/dredd) +to run a set of smoke tests to verify conformance of your app with the spec. -Automated testing tools usually require little hints in your specification to tell them how to create valid requests or what response data to expect. Dredd in particular relies on response [examples](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#exampleObject) and request parameter [x-example](http://dredd.org/en/latest/how-to-guides.html#example-values-for-request-parameters) fields. Extending your API spec with examples is a good thing on its own, since developers consuming your API will find them useful too. +Automated testing tools usually require little hints in your specification +to tell them how to create valid requests or what response data to expect. +Dredd in particular relies on response [examples](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#exampleObject) +and request parameter [x-example](http://dredd.org/en/latest/how-to-guides.html#example-values-for-request-parameters) +fields. Extending your API spec with examples is a good thing on its own, since +developers consuming your API will find them useful too. Here is an example showing how to run Dredd to test your API against the spec: @@ -536,11 +705,19 @@ describe('API (acceptance)', () => { }); ``` -The user experience is not as great as we would like it, we are looking into better solutions; see [GitHub issue #644](https://github.com/strongloop/loopback-next/issues/644). Let us know if you can recommend one! +The user experience is not as great as we would like it, we are looking into +better solutions; see [GitHub issue #644](https://github.com/strongloop/loopback-next/issues/644). +Let us know if you can recommend one! ### Test your individual REST API endpoints -You should have at least one acceptance (end-to-end) test for each of your REST API endpoints. Consider adding more tests if your endpoint depends on (custom) sequence actions to modify the behavior when the corresponding controller method is invoked via REST, compared to behavior observed when the controller method is invoked directly via JavaScript/TypeScript API. For example, if your endpoint returns different response to regular users and to admin users, then you should have two tests: one test for each user role. +You should have at least one acceptance (end-to-end) test for each of your +REST API endpoints. Consider adding more tests if your endpoint depends on +(custom) sequence actions to modify the behavior when the corresponding +controller method is invoked via REST, compared to behavior observed when +the controller method is invoked directly via JavaScript/TypeScript API. +For example, if your endpoint returns different response to regular users +and to admin users, then you should have two tests: one test for each user role. Here is an example of an acceptance test: @@ -598,4 +775,9 @@ describe('Product (acceptance)', () => { ### Test Sequence customizations -Custom sequence behavior is best tested by observing changes in behavior of affected endpoints. For example, if your sequence has an authentication step that rejects anonymous requests for certain endpoints, then you can write a test making an anonymous request to such an endpoint to verify that it's correctly rejected. These tests are essentially the same as the tests verifying implementation of individual endpoints as described in the previous section. +Custom sequence behavior is best tested by observing changes in behavior of +affected endpoints. For example, if your sequence has an authentication step +that rejects anonymous requests for certain endpoints, then you can write a test +making an anonymous request to such an endpoint to verify that it's correctly +rejected. These tests are essentially the same as the tests verifying +implementation of individual endpoints as described in the previous section.