Document RESTful services by combining hand-written documentation with auto-generated snippets produced with Spring MVC Test.

Introduction

The aim of Spring REST Docs is to help you to produce documentation for your RESTful services that is accurate and readable.

Writing high-quality documentation is difficult. One way to ease that difficulty is to use tools that are well-suited to the job. To this end, Spring REST Docs uses Asciidoctor by default. Asciidoctor processes plain text and produces HTML, styled and layed out to suit your needs. If you prefer, Spring REST Docs can also be configured to use Markdown.

Spring REST Docs makes use of snippets produced by tests written with Spring MVC Test or REST Assured. This test-driven approach helps to guarantee the accuracy of your service’s documentation. If a snippet is incorrect the test that produces it will fail.

Documenting a RESTful service is largely about describing its resources. Two key parts of each resource’s description are the details of the HTTP requests that it consumes and the HTTP responses that it produces. Spring REST Docs allows you to work with these resources and the HTTP requests and responses, shielding your documentation from the inner-details of your service’s implementation. This separation helps you to document your service’s API rather than its implementation. It also frees you to evolve the implementation without having to rework the documentation.

Getting started

This section describes how to get started with Spring REST Docs.

Sample applications

If you want to jump straight in, a number of sample applications are available:

Sample Build system Description

REST Assured

Gradle

Demonstrates the use of Spring REST Docs with REST Assured.

Slate

Gradle

Demonstrates the use of Spring REST Docs with Markdown and Slate.

Spring Data REST

Maven

Demonstrates the creation of a getting started guide and an API guide for a service implemented using Spring Data REST.

Spring HATEOAS

Gradle

Demonstrates the creation of a getting started guide and an API guide for a service implemented using Spring HATEOAS.

TestNG

Gradle

Demonstrates the use of Spring REST Docs with TestNG.

Build configuration

The first step in using Spring REST Docs is to configure your project’s build. The Spring HATEOAS and Spring Data REST samples contain a build.gradle and pom.xml respectively that you may wish to use as a reference. The key parts of the configuration are described below.

Maven
<dependency> (1)
    <groupId>org.springframework.restdocs</groupId>
    <artifactId>spring-restdocs-mockmvc</artifactId>
    <version>1.1.0.RC1</version>
    <scope>test</scope>
</dependency>

<properties> (2)
    <snippetsDirectory>${project.build.directory}/generated-snippets</snippetsDirectory>
</properties>

<build>
    <plugins>
        <plugin> (3)
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-surefire-plugin</artifactId>
            <configuration>
                <includes>
                    <include>**/*Documentation.java</include>
                </includes>
            </configuration>
        </plugin>
        <plugin> (4)
            <groupId>org.asciidoctor</groupId>
            <artifactId>asciidoctor-maven-plugin</artifactId>
            <version>1.5.2</version>
            <executions>
                <execution>
                    <id>generate-docs</id>
                    <phase>prepare-package</phase> (6)
                    <goals>
                        <goal>process-asciidoc</goal>
                    </goals>
                    <configuration>
                        <backend>html</backend>
                        <doctype>book</doctype>
                        <attributes>
                            <snippets>${snippetsDirectory}</snippets> (5)
                        </attributes>
                    </configuration>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>
1 Add a dependency on spring-restdocs-mockmvc in the test scope. If you want to use REST Assured rather than MockMvc, add a dependency on spring-restdocs-restassured instead.
2 Configure a property to define the output location for generated snippets.
3 Add the SureFire plugin and configure it to include files whose names end with Documentation.java.
4 Add the Asciidoctor plugin
5 Define an attribute named snippets that can be used when including the generated snippets in your documentation.
6 Using prepare-package allows the documentation to be included in the package.
Gradle
plugins { (1)
    id "org.asciidoctor.convert" version "1.5.2"
}

dependencies { (2)
    testCompile 'org.springframework.restdocs:spring-restdocs-mockmvc:1.1.0.RC1'
}

ext { (3)
    snippetsDir = file('build/generated-snippets')
}

test { (4)
    outputs.dir snippetsDir
}

asciidoctor { (5)
    attributes 'snippets': snippetsDir (6)
    inputs.dir snippetsDir (7)
    dependsOn test (8)
}
1 Apply the Asciidoctor plugin.
2 Add a dependency on spring-restdocs-mockmvc in the testCompile configuration. If you want to use REST Assured rather than MockMvc, add a dependency on spring-restdocs-restassured instead.
3 Configure a property to define the output location for generated snippets.
4 Configure the test task to add the snippets directory as an output.
5 Configure the asciidoctor task
6 Define an attribute named snippets that can be used when including the generated snippets in your documentation.
7 Configure the snippets directory as an input.
8 Make the task depend on the test task so that the tests are run before the documentation is created.

Packaging the documentation

You may want to package the generated documentation in your project’s jar file, for example to have it served as static content by Spring Boot. To do so, configure your project’s build so that:

  1. The documentation is generated before the jar is built

  2. The generated documentation is included in the jar

Maven
<plugin> (1)
    <groupId>org.asciidoctor</groupId>
    <artifactId>asciidoctor-maven-plugin</artifactId>
    <!-- … -->
</plugin>
<plugin> (2)
    <artifactId>maven-resources-plugin</artifactId>
    <version>2.7</version>
    <executions>
        <execution>
            <id>copy-resources</id>
            <phase>prepare-package</phase>
            <goals>
                <goal>copy-resources</goal>
            </goals>
            <configuration> (3)
                <outputDirectory>
                    ${project.build.outputDirectory}/static/docs
                </outputDirectory>
                <resources>
                    <resource>
                        <directory>
                            ${project.build.directory}/generated-docs
                        </directory>
                    </resource>
                </resources>
            </configuration>
        </execution>
    </executions>
</plugin>
1 The existing declaration for the Asciidoctor plugin.
2 The resource plugin must be declared after the Asciidoctor plugin as they are bound to the same phase (prepare-package) and the resource plugin must run after the Asciidoctor plugin to ensure that the documentation is generated before it’s copied.
3 Copy the generated documentation into the build output’s static/docs directory, from where it will be included in the jar file.
Gradle
jar {
    dependsOn asciidoctor (1)
    from ("${asciidoctor.outputDir}/html5") { (2)
        into 'static/docs'
    }
}
1 Ensure that the documentation has been generated before the jar is built.
2 Copy the generated documentation into the jar’s static/docs directory.

Generating documentation snippets

Spring REST Docs uses Spring’s MVC Test framework or REST Assured to make requests to the service that you are documenting. It then produces documentation snippets for the request and the resulting response.

Setting up your tests

Exactly how you setup your tests depends on the test framework that you’re using. Spring REST Docs provides first-class support for JUnit. Other frameworks, such as TestNG, are also supported although slightly more setup is required.

Setting up your JUnit tests

When using JUnit, the first step in generating documentation snippets is to declare a public JUnitRestDocumentation field that’s annotated as a JUnit @Rule. The JUnitRestDocumentation rule is configured with the output directory into which generated snippets should be written. This output directory should match the snippets directory that you have configured in your build.gradle or pom.xml file.

For Maven (pom.xml that will typically be target/generated-snippets and for Gradle (build.gradle) it will typically be build/generated-snippets:

Maven
@Rule
public JUnitRestDocumentation restDocumentation =
		new JUnitRestDocumentation("target/generated-snippets");
Gradle
@Rule
public JUnitRestDocumentation restDocumentation =
		new JUnitRestDocumentation("build/generated-snippets");

Next, provide an @Before method to configure MockMvc or REST Assured:

MockMvc
private MockMvc mockMvc;

@Autowired
private WebApplicationContext context;

@Before
public void setUp() {
    this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
            .apply(documentationConfiguration(this.restDocumentation)) (1)
            .build();
}
1 The MockMvc instance is configured using a MockMvcRestDocumentationConfigurer. An instance of this class can be obtained from the static documentationConfiguration() method on org.springframework.restdocs.mockmvc.MockMvcRestDocumentation.
REST Assured
private RequestSpecification spec;

@Before
public void setUp() {
    this.spec = new RequestSpecBuilder().addFilter(
            documentationConfiguration(this.restDocumentation)) (1)
            .build();
}
1 REST Assured is configured by adding a RestAssuredRestDocumentationConfigurer as a Filter. An instance of this class can be obtained from the static documentationConfiguration() method on org.springframework.restdocs.restassured.RestAssuredRestDocumentation.

The configurer applies sensible defaults and also provides an API for customizing the configuration. Refer to the configuration section for more information.

Setting up your tests without JUnit

The configuration when JUnit is not being used is largely similar to when it is being used. This section describes the key differences. The TestNG sample also illustrates the approach.

The first difference is that ManualRestDocumentation should be used in place of JUnitRestDocumentation and there’s no need for the @Rule annotation:

Maven
private ManualRestDocumentation restDocumentation =
		new ManualRestDocumentation("target/generated-snippets");
Gradle
private ManualRestDocumentation restDocumentation =
		new ManualRestDocumentation("build/generated-snippets");

Secondly, ManualRestDocumentation.beforeTest(Class, String) must be called before each test. This can be done as part of the method that is configuring MockMVC or REST Assured:

MockMvc
private MockMvc mockMvc;

@Autowired
private WebApplicationContext context;

@BeforeMethod
public void setUp(Method method) {
    this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
            .apply(documentationConfiguration(this.restDocumentation))
            .build();
    this.restDocumentation.beforeTest(getClass(), method.getName());
}
REST Assured
private RequestSpecification spec;

@BeforeMethod
public void setUp(Method method) {
    this.spec = new RequestSpecBuilder().addFilter(
            documentationConfiguration(this.restDocumentation))
            .build();
    this.restDocumentation.beforeTest(getClass(), method.getName());
}

Lastly, ManualRestDocumentation.afterTest must be called after each test. For example, with TestNG:

@AfterMethod
public void tearDown() {
    this.restDocumentation.afterTest();
}

Invoking the RESTful service

Now that the testing framework has been configured, it can be used to invoke the RESTful service and document the request and response. For example:

MockMvc
this.mockMvc.perform(get("/").accept(MediaType.APPLICATION_JSON)) (1)
    .andExpect(status().isOk()) (2)
    .andDo(document("index")); (3)
1 Invoke the root (/) of the service and indicate that an application/json response is required.
2 Assert that the service produced the expected response.
3 Document the call to the service, writing the snippets into a directory named index that will be located beneath the configured output directory. The snippets are written by a RestDocumentationResultHandler. An instance of this class can be obtained from the static document method on org.springframework.restdocs.mockmvc.MockMvcRestDocumentation.
REST Assured
RestAssured.given(this.spec) (1)
        .accept("application/json") (2)
        .filter(document("index")) (3)
        .when().get("/") (4)
        .then().assertThat().statusCode(is(200)); (5)
1 Apply the specification that was initialised in the @Before method.
2 Indicate that an application/json response is required.
3 Document the call to the service, writing the snippets into a directory named index that will be located beneath the configured output directory. The snippets are written by a RestDocumentationFilter. An instance of this class can be obtained from the static document method on org.springframework.restdocs.restassured.RestAssuredRestDocumentation.
4 Invoke the root (/) of the service.
5 Assert that the service produce the expected response.

By default, four snippets are written:

  • <output-directory>/index/curl-request.adoc

  • <output-directory>/index/http-request.adoc

  • <output-directory>/index/http-response.adoc

  • <output-directory>/index/httpie-request.adoc

Refer to Documenting your API for more information about these and other snippets that can be produced by Spring REST Docs.

Using the snippets

The generated snippets can be included in your documentation using the include macro. The snippets attribute specified in the build configuration can be used to reference the snippets output directory. For example:

include::{snippets}/index/curl-request.adoc[]

Documenting your API

This section provides more details about using Spring REST Docs to document your API.

Hypermedia

Spring REST Docs provides support for documenting the links in a Hypermedia-based API:

MockMvc
this.mockMvc.perform(get("/").accept(MediaType.APPLICATION_JSON))
    .andExpect(status().isOk())
    .andDo(document("index", links( (1)
            linkWithRel("alpha").description("Link to the alpha resource"), (2)
            linkWithRel("bravo").description("Link to the bravo resource")))); (3)
1 Configure Spring REST docs to produce a snippet describing the response’s links. Uses the static links method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.
2 Expect a link whose rel is alpha. Uses the static linkWithRel method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.
3 Expect a link whose rel is bravo.
REST Assured
RestAssured.given(this.spec)
    .accept("application/json")
    .filter(document("index", links( (1)
            linkWithRel("alpha").description("Link to the alpha resource"), (2)
            linkWithRel("bravo").description("Link to the bravo resource")))) (3)
    .get("/").then().assertThat().statusCode(is(200));
1 Configure Spring REST docs to produce a snippet describing the response’s links. Uses the static links method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.
2 Expect a link whose rel is alpha. Uses the static linkWithRel method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.
3 Expect a link whose rel is bravo.

The result is a snippet named links.adoc that contains a table describing the resource’s links.

If a link in the response has a title, the description can be omitted from its descriptor and the title will be used. If you omit the description and the link does not have a title a failure will occur.

When documenting links, the test will fail if an undocumented link is found in the response. Similarly, the test will also fail if a documented link is not found in the response and the link has not been marked as optional.

If you do not want to document a link, you can mark it as ignored. This will prevent it from appearing in the generated snippet while avoiding the failure described above.

Two link formats are understood by default:

  • Atom – links are expected to be in an array named links. Used by default when the content type of the response is compatible with application/json.

  • HAL – links are expected to be in a map named _links. Used by default when the content type of the response is compatible with application/hal+json.

If you are using Atom or HAL-format links but with a different content type you can provide one of the built-in LinkExtractor implementations to links. For example:

MockMvc
.andDo(document("index", links(halLinks(), (1)
        linkWithRel("alpha").description("Link to the alpha resource"),
        linkWithRel("bravo").description("Link to the bravo resource"))));
1 Indicate that the links are in HAL format. Uses the static halLinks method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.
REST Assured
.filter(document("index", links(halLinks(), (1)
        linkWithRel("alpha").description("Link to the alpha resource"),
        linkWithRel("bravo").description("Link to the bravo resource"))))
1 Indicate that the links are in HAL format. Uses the static halLinks method on org.springframework.restdocs.hypermedia.HypermediaDocumentation.

If your API represents its links in a format other than Atom or HAL, you can provide your own implementation of the LinkExtractor interface to extract the links from the response.

Rather than documenting links that are common to every response, such as _self and curies when using HAL, you may want to document them once in an overview section and then ignore them in the rest of your API’s documentation. To do so, you can build on the support for reusing snippets to add link descriptors to a snippet that’s preconfigured to ignore certain links. For example:

public static LinksSnippet links(LinkDescriptor... descriptors) {
    return HypermediaDocumentation.links(linkWithRel("_self").ignored().optional(),
            linkWithRel("curies").ignored()).and(descriptors);
}

Request and response payloads

In addition to the hypermedia-specific support described above, support for general documentation of request and response payloads is also provided. For example:

MockMvc
this.mockMvc.perform(get("/user/5").accept(MediaType.APPLICATION_JSON))
    .andExpect(status().isOk())
    .andDo(document("index", responseFields( (1)
            fieldWithPath("contact").description("The user's contact details"), (2)
            fieldWithPath("contact.email").description("The user's email address")))); (3)
1 Configure Spring REST docs to produce a snippet describing the fields in the response payload. To document a request requestFields can be used. Both are static methods on org.springframework.restdocs.payload.PayloadDocumentation.
2 Expect a field with the path contact. Uses the static fieldWithPath method on org.springframework.restdocs.payload.PayloadDocumentation.
3 Expect a field with the path contact.email.
REST Assured
RestAssured.given(this.spec).accept("application/json")
    .filter(document("user", responseFields( (1)
            fieldWithPath("contact").description("The user's contact details"), (2)
            fieldWithPath("contact.email").description("The user's email address")))) (3)
    .when().get("/user/5")
    .then().assertThat().statusCode(is(200));
1 Configure Spring REST docs to produce a snippet describing the fields in the response payload. To document a request requestFields can be used. Both are static methods on org.springframework.restdocs.payload.PayloadDocumentation.
2 Expect a field with the path contact. Uses the static fieldWithPath method on org.springframework.restdocs.payload.PayloadDocumentation.
3 Expect a field with the path contact.email.

The result is a snippet that contains a table describing the fields. For requests this snippet is named request-fields.adoc. For responses this snippet is named response-fields.adoc.

When documenting fields, the test will fail if an undocumented field is found in the payload. Similarly, the test will also fail if a documented field is not found in the payload and the field has not been marked as optional. For payloads with a hierarchical structure, documenting a field is sufficient for all of its descendants to also be treated as having been documented.

If you do not want to document a field, you can mark it as ignored. This will prevent it from appearing in the generated snippet while avoiding the failure described above.

By default, Spring REST Docs will assume that the payload you are documenting is JSON. If you want to document an XML payload the content type of the request or response must be compatible with application/xml.

JSON payloads

JSON field paths

JSON field paths use either dot notation or bracket notation. Dot notation uses '.' to separate each key in the path; a.b, for example. Bracket notation wraps each key in square brackets and single quotes; ['a']['b'], for example. In either case, [] is used to identify an array. Dot notation is more concise, but using bracket notation enables the use of . within a key name; ['a.b'], for example. The two different notations can be used in the same path; a['b'], for example.

With this JSON payload:

{
    "a":{
        "b":[
            {
                "c":"one"
            },
            {
                "c":"two"
            },
            {
                "d":"three"
            }
        ],
        "e.dot" : "four"
    }
}

The following paths are all present:

Path Value

a

An object containing b

a.b

An array containing three objects

['a']['b']

An array containing three objects

a['b']

An array containing three objects

['a'].b

An array containing three objects

a.b[]

An array containing three objects

a.b[].c

An array containing the strings one and two

a.b[].d

The string three

a['e.dot']

The string four

['a']['e.dot']

The string four

A response that uses an array at its root can also be documented. The path [] will refer to the entire array. You can then use bracket or dot notation to identify fields within the array’s entries. For example, [].id corresponds to the id field of every object found in the following array:

[
    {
        "id":1
    },
    {
        "id":2
    }
]
JSON field types

When a field is documented, Spring REST Docs will attempt to determine its type by examining the payload. Seven different types are supported:

Type Description

array

The value of each occurrence of the field is an array

boolean

The value of each occurrence of the field is a boolean (true or false)

object

The value of each occurrence of the field is an object

number

The value of each occurrence of the field is a number

null

The value of each occurrence of the field is null

string

The value of each occurrence of the field is a string

varies

The field occurs multiple times in the payload with a variety of different types

The type can also be set explicitly using the type(Object) method on FieldDescriptor. The result of the supplied Object's toString method will be used in the documentation. Typically, one of the values enumerated by JsonFieldType will be used:

MockMvc
.andDo(document("index", responseFields(
        fieldWithPath("contact.email")
                .type(JsonFieldType.STRING) (1)
                .description("The user's email address"))));
1 Set the field’s type to string.
REST Assured
.filter(document("user", responseFields(
        fieldWithPath("contact.email")
                .type(JsonFieldType.STRING) (1)
                .description("The user's email address"))))
1 Set the field’s type to string.

XML payloads

XML field paths

XML field paths are described using XPath. / is used to descend into a child node.

XML field types

When documenting an XML payload, you must provide a type for the field using the type(Object) method on FieldDescriptor. The result of the supplied type’s toString method will be used in the documentation.

Reusing field descriptors

In addition to the general support for reusing snippets, the request and response snippets allow additional descriptors to be configured with a path prefix. This allows the descriptors for a repeated portion of a request or response payload to be created once and then reused.

Consider an endpoint that returns a book:

{
    "title": "Pride and Prejudice",
    "author": "Jane Austen"
}

The paths for title and author are simply title and author respectively.

Now consider an endpoint that returns an array of books:

[{
    "title": "Pride and Prejudice",
    "author": "Jane Austen"
},
{
    "title": "To Kill a Mockingbird",
    "author": "Harper Lee"
}]

The paths for title and author are [].title and [].author respectively. The only difference between the single book and the array of books is that the fields' paths now have a []. prefix.

The descriptors that document a book can be created:

FieldDescriptor[] book = new FieldDescriptor[] {
        fieldWithPath("title").description("Title of the book"),
        fieldWithPath("author").description("Author of the book") };

They can then be used to document a single book:

MockMvc
this.mockMvc.perform(get("/books/1").accept(MediaType.APPLICATION_JSON))
        .andExpect(status().isOk())
        .andDo(document("book", responseFields(book))); (1)
1 Document title and author using existing descriptors
REST Assured
RestAssured.given(this.spec).accept("application/json")
    .filter(document("book", responseFields(book))) (1)
    .when().get("/books/1")
    .then().assertThat().statusCode(is(200));
1 Document title and author using existing descriptors

And an array of books:

MockMvc
this.mockMvc.perform(get("/books").accept(MediaType.APPLICATION_JSON))
        .andExpect(status().isOk())
        .andDo(document("book", responseFields(
                fieldWithPath("[]").description("An array of books")) (1)
                .andWithPrefix("[].", book))); (2)
1 Document the array
2 Document [].title and [].author using the existing descriptors prefixed with [].
REST Assured
RestAssured.given(this.spec).accept("application/json")
    .filter(document("books", responseFields(
        fieldWithPath("[]").description("An array of books")) (1)
        .andWithPrefix("[].", book))) (2)
.when().get("/books/1")
.then().assertThat().statusCode(is(200));
1 Document the array
2 Document [].title and [].author using the existing descriptors prefixed with [].

Request parameters

A request’s parameters can be documented using requestParameters. Request parameters can be included in a GET request’s query string. For example:

MockMvc
this.mockMvc.perform(get("/users?page=2&per_page=100")) (1)
    .andExpect(status().isOk())
    .andDo(document("users", requestParameters( (2)
            parameterWithName("page").description("The page to retrieve"), (3)
            parameterWithName("per_page").description("Entries per page") (4)
    )));
1 Perform a GET request with two parameters, page and per_page in the query string.
2 Configure Spring REST Docs to produce a snippet describing the request’s parameters. Uses the static requestParameters method on org.springframework.restdocs.request.RequestDocumentation.
3 Document the page parameter. Uses the static parameterWithName method on org.springframework.restdocs.request.RequestDocumentation.
4 Document the per_page parameter.
REST Assured
RestAssured.given(this.spec)
    .filter(document("users", requestParameters( (1)
            parameterWithName("page").description("The page to retrieve"), (2)
            parameterWithName("per_page").description("Entries per page")))) (3)
    .when().get("/users?page=2&per_page=100") (4)
    .then().assertThat().statusCode(is(200));
1 Configure Spring REST Docs to produce a snippet describing the request’s parameters. Uses the static requestParameters method on org.springframework.restdocs.request.RequestDocumentation.
2 Document the page parameter. Uses the static parameterWithName method on org.springframework.restdocs.request.RequestDocumentation.
3 Document the per_page parameter.
4 Perform a GET request with two parameters, page and per_page in the query string.

Request parameters can also be included as form data in the body of a POST request:

MockMvc
this.mockMvc.perform(post("/users").param("username", "Tester")) (1)
    .andExpect(status().isCreated())
    .andDo(document("create-user", requestParameters(
            parameterWithName("username").description("The user's username")
    )));
1 Perform a POST request with a single parameter, username.
REST Assured
RestAssured.given(this.spec)
    .filter(document("create-user", requestParameters(
            parameterWithName("username").description("The user's username"))))
    .formParam("username", "Tester") (1)
    .when().post("/users") (2)
    .then().assertThat().statusCode(is(200));
1 Configure the username parameter.
2 Perform the POST request.

In both cases, the result is a snippet named request-parameters.adoc that contains a table describing the parameters that are supported by the resource.

When documenting request parameters, the test will fail if an undocumented request parameter is used in the request. Similarly, the test will also fail if a documented request parameter is not found in the request and the request parameter has not been marked as optional.

If you do not want to document a request parameter, you can mark it as ignored. This will prevent it from appearing in the generated snippet while avoiding the failure described above.

Path parameters

A request’s path parameters can be documented using pathParameters. For example:

MockMvc
this.mockMvc.perform(get("/locations/{latitude}/{longitude}", 51.5072, 0.1275)) (1)
    .andExpect(status().isOk())
    .andDo(document("locations", pathParameters( (2)
            parameterWithName("latitude").description("The location's latitude"), (3)
            parameterWithName("longitude").description("The location's longitude") (4)
    )));
1 Perform a GET request with two path parameters, latitude and longitude.
2 Configure Spring REST Docs to produce a snippet describing the request’s path parameters. Uses the static pathParameters method on org.springframework.restdocs.request.RequestDocumentation.
3 Document the parameter named latitude. Uses the static parameterWithName method on org.springframework.restdocs.request.RequestDocumentation.
4 Document the parameter named longitude.
REST Assured
RestAssured.given(this.spec)
    .filter(document("locations", pathParameters( (1)
            parameterWithName("latitude").description("The location's latitude"), (2)
            parameterWithName("longitude").description("The location's longitude")))) (3)
    .when().get("/locations/{latitude}/{longitude}", 51.5072, 0.1275) (4)
    .then().assertThat().statusCode(is(200));
1 Configure Spring REST Docs to produce a snippet describing the request’s path parameters. Uses the static pathParameters method on org.springframework.restdocs.request.RequestDocumentation.
2 Document the parameter named latitude. Uses the static parameterWithName method on org.springframework.restdocs.request.RequestDocumentation.
3 Document the parameter named longitude.
4 Perform a GET request with two path parameters, latitude and longitude.

The result is a snippet named path-parameters.adoc that contains a table describing the path parameters that are supported by the resource.

To make the path parameters available for documentation, the request must be built using one of the methods on RestDocumentationRequestBuilders rather than MockMvcRequestBuilders.

When documenting path parameters, the test will fail if an undocumented path parameter is used in the request. Similarly, the test will also fail if a documented path parameter is not found in the request and the path parameter has not been marked as optional.

If you do not want to document a path parameter, you can mark it as ignored. This will prevent it from appearing in the generated snippet while avoiding the failure described above.

HTTP headers

The headers in a request or response can be documented using requestHeaders and responseHeaders respectively. For example:

MockMvc
this.mockMvc
    .perform(get("/people").header("Authorization", "Basic dXNlcjpzZWNyZXQ=")) (1)
    .andExpect(status().isOk())
    .andDo(document("headers",
            requestHeaders( (2)
                    headerWithName("Authorization").description(
                            "Basic auth credentials")), (3)
            responseHeaders( (4)
                    headerWithName("X-RateLimit-Limit").description(
                            "The total number of requests permitted per period"),
                    headerWithName("X-RateLimit-Remaining").description(
                            "Remaining requests permitted in current period"),
                    headerWithName("X-RateLimit-Reset").description(
                            "Time at which the rate limit period will reset"))));
1 Perform a GET request with an Authorization header that uses basic authentication
2 Configure Spring REST Docs to produce a snippet describing the request’s headers. Uses the static requestHeaders method on org.springframework.restdocs.headers.HeaderDocumentation.
3 Document the Authorization header. Uses the static headerWithName method on `org.springframework.restdocs.headers.HeaderDocumentation.
4 Produce a snippet describing the response’s headers. Uses the static responseHeaders method on org.springframework.restdocs.headers.HeaderDocumentation.
REST Assured
RestAssured.given(this.spec)
    .filter(document("headers",
            requestHeaders( (1)
                    headerWithName("Authorization").description(
                            "Basic auth credentials")), (2)
            responseHeaders( (3)
                    headerWithName("X-RateLimit-Limit").description(
                            "The total number of requests permitted per period"),
                    headerWithName("X-RateLimit-Remaining").description(
                            "Remaining requests permitted in current period"),
                    headerWithName("X-RateLimit-Reset").description(
                            "Time at which the rate limit period will reset"))))
    .header("Authroization", "Basic dXNlcjpzZWNyZXQ=") (4)
    .when().get("/people")
    .then().assertThat().statusCode(is(200));
1 Configure Spring REST Docs to produce a snippet describing the request’s headers. Uses the static requestHeaders method on org.springframework.restdocs.headers.HeaderDocumentation.
2 Document the Authorization header. Uses the static headerWithName method on `org.springframework.restdocs.headers.HeaderDocumentation.
3 Produce a snippet describing the response’s headers. Uses the static responseHeaders method on org.springframework.restdocs.headers.HeaderDocumentation.
4 Configure the request with an Authorization header that uses basic authentication

The result is a snippet named request-headers.adoc and a snippet named response-headers.adoc. Each contains a table describing the headers.

When documenting HTTP Headers, the test will fail if a documented header is not found in the request or response.

Reusing snippets

It’s common for an API that’s being documented to have some features that are common across several of its resources. To avoid repetition when documenting such resources a Snippet configured with the common elements can be reused.

First, create the Snippet that describes the common elements. For example:

protected final LinksSnippet pagingLinks = links(
        linkWithRel("first").optional().description("The first page of results"),
        linkWithRel("last").optional().description("The last page of results"),
        linkWithRel("next").optional().description("The next page of results"),
        linkWithRel("prev").optional().description("The previous page of results"));

Second, use this snippet and add further descriptors that are resource-specific. For example:

MockMvc
this.mockMvc.perform(get("/").accept(MediaType.APPLICATION_JSON))
    .andExpect(status().isOk())
    .andDo(document("example", this.pagingLinks.and( (1)
            linkWithRel("alpha").description("Link to the alpha resource"),
            linkWithRel("bravo").description("Link to the bravo resource"))));
1 Reuse the pagingLinks Snippet calling and to add descriptors that are specific to the resource that is being documented.
REST Assured
RestAssured.given(this.spec)
    .accept("application/json")
    .filter(document("example", this.pagingLinks.and( (1)
            linkWithRel("alpha").description("Link to the alpha resource"),
            linkWithRel("bravo").description("Link to the bravo resource"))))
    .get("/").then().assertThat().statusCode(is(200));
1 Reuse the pagingLinks Snippet calling and to add descriptors that are specific to the resource that is being documented.

The result of the example is that links with the rels first, last, next, previous, alpha, and bravo are all documented.

Documenting constraints

Spring REST Docs provides a number of classes that can help you to document constraints. An instance of ConstraintDescriptions can be used to access descriptions of a class’s constraints. For example:

public void example() {
    ConstraintDescriptions userConstraints = new ConstraintDescriptions(UserInput.class); (1)
    List<String> descriptions = userConstraints.descriptionsForProperty("name"); (2)
}

static class UserInput {

    @NotNull
    @Size(min = 1)
    String name;

    @NotNull
    @Size(min = 8)
    String password;

}
1 Create an instance of ConstraintDescriptions for the UserInput class
2 Get the descriptions of the name property’s constraints. This list will contain two descriptions; one for the NotNull constraint and one for the Size constraint.

The ApiDocumentation class in the Spring HATEOAS sample shows this functionality in action.

Finding constraints

By default, constraints are found using a Bean Validation Validator. Currently, only property constraints are supported. You can customize the Validator that’s used by creating ConstraintDescriptions with a custom ValidatorConstraintResolver instance. To take complete control of constraint resolution, your own implementation of ConstraintResolver can be used.

Describing constraints

Default descriptions are provided for all of the Bean Validation 1.1’s constraints:

  • AssertFalse

  • AssertTrue

  • DecimalMax

  • DecimalMin

  • Digits

  • Future

  • Max

  • Min

  • NotNull

  • Null

  • Past

  • Pattern

  • Size

Default descriptions are also provided for constraints from Hibernate Validator:

  • CreditCardNumber

  • EAN

  • Email

  • Length

  • LuhnCheck

  • Mod10Check

  • Mod11Check

  • NotBlank

  • NotEmpty

  • Range

  • SafeHtml

  • URL

To override the default descriptions, or to provide a new description, create a resource bundle with the base name org.springframework.restdocs.constraints.ConstraintDescriptions. The Spring HATEOAS-based sample contains an example of such a resource bundle.

Each key in the resource bundle is the fully-qualified name of a constraint plus .description. For example, the key for the standard @NotNull constraint is javax.validation.constraints.NotNull.description.

Property placeholder’s referring to a constraint’s attributes can be used in its description. For example, the default description of the @Min constraint, Must be at least ${value}, refers to the constraint’s value attribute.

To take more control of constraint description resolution, create ConstraintDescriptions with a custom ResourceBundleConstraintDescriptionResolver. To take complete control, create ConstraintDescriptions with a custom ConstraintDescriptionResolver implementation.

Using constraint descriptions in generated snippets

Once you have a constraint’s descriptions, you’re free to use them however you like in the generated snippets. For example, you may want to include the constraint descriptions as part of a field’s description. Alternatively, you could include the constraints as extra information in the request fields snippet. The ApiDocumentation class in the Spring HATEOAS-based sample illustrates the latter approach.

Default snippets

A number of snippets are produced automatically when you document a request and response.

Snippet Description

curl-request.adoc

Contains the curl command that is equivalent to the MockMvc call that is being documented

httpie-request.adoc

Contains the HTTPie command that is equivalent to the MockMvc call that is being documented

http-request.adoc

Contains the HTTP request that is equivalent to the MockMvc call that is being documented

http-response.adoc

Contains the HTTP response that was returned

You can configure which snippets are produced by default. Please refer to the configuration section for more information.

Using parameterized output directories

The output directory used by document can be parameterized. The following parameters are supported:

Parameter Description

{methodName}

The unmodified name of the test method

{method-name}

The name of the test method, formatted using kebab-case

{method_name}

The name of the test method, formatted using snake_case

{ClassName}

The unmodified name of the test class

{class-name}

The name of the test class, formatted using kebab-case

{class_name}

The nome of the test class, formatted using snake_case

{step}

The count of calls made to the service in the current test

For example, document("{class-name}/{method-name}") in a test method named creatingANote on the test class GettingStartedDocumentation, will write snippets into a directory named getting-started-documentation/creating-a-note.

A parameterized output directory is particularly useful in combination with an @Before method. It allows documentation to be configured once in a setup method and then reused in every test in the class:

MockMvc
@Before
public void setUp() {
    this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
            .apply(documentationConfiguration(this.restDocumentation))
            .alwaysDo(document("{method-name}/{step}/")).build();
}
REST Assured
@Before
public void setUp() {
    this.spec = new RequestSpecBuilder()
            .addFilter(documentationConfiguration(this.restDocumentation))
            .addFilter(document("{method-name}/{step}")).build();
}

With this configuration in place, every call to the service you are testing will produce the default snippets without any further configuration. Take a look at the GettingStartedDocumentation classes in each of the sample applications to see this functionality in action.

Customizing the output

Customizing the generated snippets

Spring REST Docs uses Mustache templates to produce the generated snippets. Default templates are provided for each of the snippets that Spring REST Docs can produce. To customize a snippet’s content, you can provide your own template.

Templates are loaded from the classpath from an org.springframework.restdocs.templates subpackage. The name of the subpackage is determined by the ID of the template format that is in use. The default template format, Asciidoctor, has the ID asciidoctor so snippets are loaded from org.springframework.restdocs.templates.asciidoctor. Each template is named after the snippet that it will produce. For example, to override the template for the curl-request.adoc snippet, create a template named curl-request.snippet in src/test/resources/org/springframework/restdocs/templates/asciidoctor.

Including extra information

There are two ways to provide extra information for inclusion in a generated snippet:

  1. Use the attributes method on a descriptor to add one or more attributes to it.

  2. Pass in some attributes when calling curlRequest, httpRequest, httpResponse, etc. Such attributes will be associated with the snippet as a whole.

Any additional attributes are made available during the template rendering process. Coupled with a custom snippet template, this makes it possible to include extra information in a generated snippet.

A concrete example of the above is the addition of a constraints column and a title when documenting request fields. The first step is to provide a constraints attribute for each field that you are documenting and to provide a title attribute:

MockMvc
.andDo(document("create-user", requestFields(
        attributes(key("title").value("Fields for user creation")), (1)
        fieldWithPath("name").description("The user's name")
                .attributes(key("constraints")
                        .value("Must not be null. Must not be empty")), (2)
        fieldWithPath("email").description("The user's email address")
                .attributes(key("constraints")
                        .value("Must be a valid email address"))))); (3)
1 Configure the title attribute for the request fields snippet
2 Set the constraints attribute for the name field
3 Set the constraints attribute for the email field
REST Assured
.filter(document("create-user", requestFields(
        attributes(key("title").value("Fields for user creation")), (1)
        fieldWithPath("name").description("The user's name")
                .attributes(key("constraints")
                        .value("Must not be null. Must not be empty")), (2)
        fieldWithPath("email").description("The user's email address")
                .attributes(key("constraints")
                        .value("Must be a valid email address"))))) (3)
1 Configure the title attribute for the request fields snippet
2 Set the constraints attribute for the name field
3 Set the constraints attribute for the email field

The second step is to provide a custom template named request-fields.snippet that includes the information about the fields' constraints in the generated snippet’s table and adds a title:

.{{title}} (1)
|===
|Path|Type|Description|Constraints (2)

{{#fields}}
|{{path}}
|{{type}}
|{{description}}
|{{constraints}} (3)

{{/fields}}
|===
1 Add a title to the table
2 Add a new column named "Constraints"
3 Include the descriptors' constraints attribute in each row of the table

Customizing requests and responses

There may be situations where you do not want to document a request exactly as it was sent or a response exactly as it was received. Spring REST Docs provides a number of preprocessors that can be used to modify a request or response before it’s documented.

Preprocessing is configured by calling document with an OperationRequestPreprocessor, and/or an OperationResponsePreprocessor. Instances can be obtained using the static preprocessRequest and preprocessResponse methods on Preprocessors. For example:

MockMvc
this.mockMvc.perform(get("/")).andExpect(status().isOk())
    .andDo(document("index", preprocessRequest(removeHeaders("Foo")), (1)
            preprocessResponse(prettyPrint()))); (2)
1 Apply a request preprocessor that will remove the header named Foo.
2 Apply a response preprocessor that will pretty print its content.
REST Assured
RestAssured.given(this.spec)
    .filter(document("index", preprocessRequest(removeHeaders("Foo")), (1)
            preprocessResponse(prettyPrint()))) (2)
.when().get("/")
.then().assertThat().statusCode(is(200));
1 Apply a request preprocessor that will remove the header named Foo.
2 Apply a response preprocessor that will pretty print its content.

Alternatively, you may want to apply the same preprocessors to every test. You can do so by configuring the preprocessors in your @Before method and using the support for parameterized output directories:

MockMvc
private MockMvc mockMvc;

private RestDocumentationResultHandler document;

@Before
public void setup() {
    this.document = document("{method-name}", (1)
            preprocessRequest(removeHeaders("Foo")),
            preprocessResponse(prettyPrint()));
    this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
            .apply(documentationConfiguration(this.restDocumentation))
            .alwaysDo(this.document) (2)
            .build();
}
1 Create a RestDocumentationResultHandler, configured to preprocess the request and response.
2 Create a MockMvc instance, configured to always call the documentation result handler.
REST Assured
private RequestSpecification spec;

private RestDocumentationFilter document;

@Before
public void setup() {
    this.document = document("{method-name}",
            preprocessRequest(removeHeaders("Foo")),
            preprocessResponse(prettyPrint())); (1)
    this.spec = new RequestSpecBuilder()
            .addFilter(documentationConfiguration(this.restDocumentation))
            .addFilter(this.document)(2)
            .build();
}
1 Create a RestDocumentationFilter, configured to preprocess the request and response.
2 Create a RequestSpecification instance, configured to always call the documentation filter.

Then, in each test, any configuration specific to that test can be performed. For example:

MockMvc
this.document.snippets( (1)
        links(linkWithRel("self").description("Canonical self link")));
this.mockMvc.perform(get("/")) (2)
        .andExpect(status().isOk());
1 Document the links specific to the resource that is being tested
2 The request and response will be preprocessed due to the use of alwaysDo above.
REST Assured
this.document.snippets( (1)
        links(linkWithRel("self").description("Canonical self link")));
RestAssured.given(this.spec) (2)
    .when().get("/")
    .then().assertThat().statusCode(is(200));
1 Document the links specific to the resource that is being tested
2 The request and response will be preprocessed due to the configuration of the RequestSpecification in the setUp method.

Various built in preprocessors, including those illustrated above, are available via the static methods on Preprocessors. See below for further details.

Preprocessors

Pretty printing

prettyPrint on Preprocessors formats the content of the request or response to make it easier to read.

If you’re documenting a Hypermedia-based API, you may want to encourage clients to navigate the API using links rather than through the use of hard coded URIs. One way to do this is to limit the use of URIs in the documentation. maskLinks on Preprocessors replaces the href of any links in the response with …​. A different replacement can also be specified if you wish.

Removing headers

removeHeaders on Preprocessors removes any headers from the request or response where the name is equal to any of the given header names.

removeMatchingHeaders on Preprocessors removes any headers from the request or response where the name matches any of the given regular expression patterns.

Replacing patterns

replacePattern on Preprocessors provides a general purpose mechanism for replacing content in a request or response. Any occurrences of a regular expression are replaced.

Modifying request parameters

modifyParameters on Preprocessors can be used to add, set, and remove request parameters.

Modifying URIs

If you are using MockMvc, URIs should be customized by changing the configuration.

modifyUris on RestAssuredPreprocessors can be used to modify any URIs in a request or a response. When using REST Assured, this allows you to customize the URIs that appear in the documentation while testing a local instance of the service.

Writing your own preprocessor

If one of the built-in preprocessors does not meet your needs, you can write your own by implementing the OperationPreprocessor interface. You can then use your custom preprocessor in exactly the same way as any of the built-in preprocessors.

If you only want to modify the content (body) of a request or response, consider implementing the ContentModifier interface and using it with the built-in ContentModifyingOperationPreprocessor.

Configuration

Documented URIs

As REST Assured tests a service by making actual HTTP requests, the documented URIs cannot be customized in this way. You should use the REST-Assured specific preprocessor instead.

When using MockMvc, the default configuration for URIs documented by Spring REST Docs is:

Setting Default

Scheme

http

Host

localhost

Port

8080

This configuration is applied by MockMvcRestDocumentationConfigurer. You can use its API to change one or more of the defaults to suit your needs:

this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
        .apply(documentationConfiguration(this.restDocumentation).uris()
                .withScheme("https")
                .withHost("example.com")
                .withPort(443))
        .build();
To configure a request’s context path, use the contextPath method on MockHttpServletRequestBuilder.

Snippet encoding

The default snippet encoding is UTF-8. You can change the default snippet encoding using the RestDocumentationConfigurer API. For example, to use ISO-8859-1:

MockMvc
this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
        .apply(documentationConfiguration(this.restDocumentation)
                .snippets().withEncoding("ISO-8859-1"))
        .build();
REST Assured
this.spec = new RequestSpecBuilder()
        .addFilter(documentationConfiguration(this.restDocumentation)
                .snippets().withEncoding("ISO-8859-1"))
        .build();
When Spring REST Docs converts a request or response’s content to a String, the charset specified in the Content-Type header will be used if it is available. In its absence, the JVM’s default Charset will be used. The JVM’s default Charset can be configured using the file.encoding system property.

Snippet template format

The default snippet template format is Asciidoctor. Markdown is also supported out of the box. You can change the default format using the RestDocumentationConfigurer API:

MockMvc
this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
        .apply(documentationConfiguration(this.restDocumentation)
                .snippets().withTemplateFormat(TemplateFormats.markdown()))
        .build();
REST Assured
this.spec = new RequestSpecBuilder()
        .addFilter(documentationConfiguration(this.restDocumentation)
                .snippets().withTemplateFormat(TemplateFormats.markdown()))
        .build();

Default snippets

Four snippets are produced by default:

  • curl-request

  • http-request

  • http-response

  • httpie-request

You can change the default snippet configuration during setup using the RestDocumentationConfigurer API. For example, to only produce the curl-request snippet by default:

MockMvc
this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context)
        .apply(documentationConfiguration(this.restDocumentation).snippets()
                .withDefaults(curlRequest()))
        .build();
REST Assured
this.spec = new RequestSpecBuilder()
        .addFilter(documentationConfiguration(this.restDocumentation).snippets()
                .withDefaults(curlRequest()))
        .build();

Working with Asciidoctor

This section describes any aspects of working with Asciidoctor that are particularly relevant to Spring REST Docs.

Including snippets

The include macro is used to include generated snippets in your documentation. The snippets attribute specified in the build configuration can be used to reference the snippets output directory, for example:

include::{snippets}/index/curl-request.adoc[]

Customizing tables

Many of the snippets contain a table in its default configuration. The appearance of the table can be customized, either by providing some additional configuration when the snippet is included or by using a custom snippet template.

Formatting columns

Asciidoctor has rich support for formatting a table’s columns. For example, the widths of a table’s columns can be specified using the cols attribute:

[cols=1,3] (1)
include::{snippets}/index/links.adoc[]
1 The table’s width will be split across its two columns with the second column being three times as wide as the first.

Configuring the title

The title of a table can be specified using a line prefixed by a .:

.Links (1)
include::{snippets}/index/links.adoc[]
1 The table’s title will be Links.

Further reading

Refer to the Tables section of the Asciidoctor user manual for more information about customizing tables.

Working with Markdown

This section describes any aspects of working with Markdown that are particularly relevant to Spring REST Docs.

Limitations

Markdown was originally designed for people writing for the web and, as such, isn’t as well-suited to writing documentation as Asciidoctor. Typically, these limitations are overcome by using another tool that builds on top of Markdown.

Markdown has no official support for tables. Spring REST Docs' default Markdown snippet templates use Markdown Extra’s table format.

Including snippets

Markdown has no built-in support for including one Markdown file in another. To include the generated snippets of Markdown in your documentation, you should use an additional tool that supports this functionality. One example that’s particularly well-suited to documenting APIs is Slate.

Contributing

Spring REST Docs is intended to make it easy for you to produce high-quality documentation for your RESTful services. However, we can’t achieve that goal without your contributions.

Questions

You can ask questions about Spring REST Docs on StackOverflow using the spring-restdocs tag. Similarly, we encourage you to help your fellow Spring REST Docs users by answering questions.

Bugs

If you believe you have found a bug, please take a moment to search the existing issues. If no one else has reported the problem, please open a new issue that describes the problem in detail and, ideally, includes a test that reproduces it.

Enhancements

If you’d like an enhancement to be made to Spring REST Docs, pull requests are most welcome. The source code is on GitHub. You may want to search the existing issues and pull requests to see if the enhancement is already being worked on. You may also want to open a new issue to discuss a possible enhancement before work on it begins.