Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Test Driven Approaches to Documenting RESTful APIs

jlstrater
February 09, 2016

Test Driven Approaches to Documenting RESTful APIs

Documentation generated from source code is very popular. Solutions such as Swagger are available for many different languages and frameworks. However, limitations of annotation based tools are becoming apparent. An overwhelming number of documentation annotations make for great docs but muddy the source code. Then, something changes and the docs are out of date again.

That is where test-driven approaches come in.

Test-driven documentation solutions, such as Spring Rest Docs, generate example snippets for requests and responses from tests ensuring both code coverage and accurate documentation. It can even fail the build when documentation becomes out of date. This session will walk through how to implement test-driven documentation solutions for groovy ecosystem technologies like Spring Boot. Attendees should have a basic understanding of AsciiDoc and how to construct RESTful APIs in Spring Boot or Grails.

jlstrater

February 09, 2016
Tweet

More Decks by jlstrater

Other Decks in Technology

Transcript

  1. Background • Creating RESTful APIs with Groovy • Spring Boot

    • Grails • Documentation • Swagger • Asciidoctor
  2. Custom JSON {
 "swagger": "2.0",
 "info": {
 "version": "1",
 "title":

    "My Service",
 "contact": {
 "name": "Company Name"
 },
 "license": {}
 },
 "host": "example.com",
 "basepath": "/docs",
 "tags": [
 {
 "name": "group name",
 "description": "what this group of api endpoints is for",
 }
 ],
 "paths": {
 "/api/v1/groupname/resource":
 .
 .
 .
 }
  3. public class Swagger2MarkupTest {
 @Autowired
 private WebApplicationContext context;
 
 private

    MockMvc mockMvc;
 
 @Before
 public void setUp() {
 this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context).build();
 }
 

  4. public class Swagger2MarkupTest {
 @Autowired
 private WebApplicationContext context;
 
 private

    MockMvc mockMvc;
 
 @Before
 public void setUp() {
 this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context).build();
 }
 
 @Test
 public void convertSwaggerToAsciiDoc() throws Exception {
 this.mockMvc.perform(get("/v2/api-docs")
 .accept(MediaType.APPLICATION_JSON))
 .andDo(Swagger2MarkupResultHandler.outputDirectory(“src/docs/ asciidoc/generated").build())
 .andExpect(status().isOk());
 }
 

  5. public class Swagger2MarkupTest {
 @Autowired
 private WebApplicationContext context;
 
 private

    MockMvc mockMvc;
 
 @Before
 public void setUp() {
 this.mockMvc = MockMvcBuilders.webAppContextSetup(this.context).build();
 }
 
 @Test
 public void convertSwaggerToAsciiDoc() throws Exception {
 this.mockMvc.perform(get("/v2/api-docs")
 .accept(MediaType.APPLICATION_JSON))
 .andDo(Swagger2MarkupResultHandler.outputDirectory(“src/docs/ asciidoc/generated").build())
 .andExpect(status().isOk());
 }
 
 @Test
 public void convertSwaggerToMarkdown() throws Exception {
 this.mockMvc.perform(get("/v2/api-docs")
 .accept(MediaType.APPLICATION_JSON))
 .andDo(Swagger2MarkupResultHandler.outputDirectory(“src/docs/ markdown/generated")
 .withMarkupLanguage(MarkupLanguage.MARKDOWN).build())
 .andExpect(status().isOk());
 }
 }
  6. Game Changers • Generated code snippets • Tests fail when

    documentation is missing or out-of- date • Supports Level III Rest APIs (Hypermedia)
  7. Getting Started • Documentation - Spring Projects • Documenting RESTful

    Apis - SpringOne2GX 2015 - Andy Wilkinson • Spring REST Docs - Documenting RESTful APIs using your tests - Devoxx Belgium 2015 - Anders Evers
  8. Groovier Spring REST docs • Groovy Spring Boot Project •

    Level II Rest API + Asciidoctor Gradle plugin
  9. Groovier Spring REST docs • Groovy Spring Boot Project •

    Level II Rest API + Asciidoctor Gradle plugin + MockMVC and documentation to Spock tests
  10. Groovy Spring Boot App • Start with lazybones spring boot

    app • Add mock endpoints for example https://github.com/jlstrater/groovy-spring-boot- restdocs-example
  11. Endpoints @CompileStatic
 @RestController
 @RequestMapping('/hello')
 @Slf4j
 class ExampleController {
 
 @RequestMapping(method

    = RequestMethod.GET, produces = 'application/json', consumes = 'application/json')
 Example list() {
 new Example(name: 'World')
 }
 
 @RequestMapping(method = RequestMethod.POST, produces = 'application/json', consumes = 'application/json')
 Example list(@RequestBody Example example) {
 example
 }
 }
  12. Install classpath 'org.asciidoctor:asciidoctor-gradle-plugin:1.5.3'
 apply plugin: 'org.asciidoctor.convert'
 
 asciidoctor {
 sourceDir

    = file('src/docs')
 outputDir "$projectDir/src/main/resources/public"
 backends 'html5'
 attributes 'source-highlighter' : 'prettify',
 'imagesdir':'images',
 'toc':'left',
 'icons': 'font',
 'setanchors':'true',
 'idprefix':'',
 'idseparator':'-',
 'docinfo1':'true'
 }

  13. Example AsciiDoc = Gr8Data API Guide
 Jenn Strater;
 :doctype: book


    :icons: font
 :source-highlighter: highlightjs
 :toc: left
 :toclevels: 4
 :sectlinks:
 
 [introduction]
 = Introduction
 
 The Gr8Data API is a RESTful web service for aggregating and displaying gender ratios from
 various companies across the world. This document outlines how to submit data from your company or team and
 how to access the aggregate data.
 
 [[overview-http-verbs]]
 == HTTP verbs
 
 The Gr8Data API tries to adhere as closely as possible to standard HTTP and REST conventions in its
 use of HTTP verbs.
  14. Stand Alone Setup class ExampleControllerSpec extends Specification {
 protected MockMvc

    mockMvc
 
 void setup() {
 this.mockMvc = MockMvcBuilders.standaloneSetup( new ExampleController()).build()
 }
 
 void 'test and document get with example endpoint’() {
 when:
 ResultActions result = this.mockMvc.perform( get(‘/hello’).contentType(MediaType.APPLICATION_JSON)) then:
 result .andExpect(status().isOk()) .andExpect(jsonPath("name").value("World"))
 .andExpect(jsonPath("message").value("Hello, World!”)) }
  15. Web Context Setup void setup() {
 this.mockMvc = MockMvcBuilders .webAppContextSetup(this.context)

    .build()
 } If context is null, remember to add spock-spring!!
  16. Gradle asciidoctor {
 dependsOn test
 sourceDir = file('src/docs')
 outputDir "$projectDir/src/main/resources/public"


    + inputs.dir snippetsDir
 backends 'html5'
 attributes 'source-highlighter' : 'prettify',
 'imagesdir':'images',
 'toc':'left',
 'icons': 'font',
 'setanchors':'true',
 'idprefix':'',
 'idseparator':'-',
 'docinfo1':'true',
 + 'snippets': snippetsDir,
 }
  17. Setup & GET class ExampleControllerSpec extends Specification {
 
 @Rule


    RestDocumentation restDocumentation = new RestDocumentation('src/docs/generated- snippets')
 
 protected MockMvc mockMvc
 
 void setup() {
 this.mockMvc = MockMvcBuilders.standaloneSetup(new ExampleController())
 .apply(documentationConfiguration(this.restDocumentation))
 .build()
 }
  18. Setup & GET class ExampleControllerSpec extends Specification {
 
 @Rule


    RestDocumentation restDocumentation = new RestDocumentation('src/docs/generated- snippets')
 
 protected MockMvc mockMvc
 
 void setup() {
 this.mockMvc = MockMvcBuilders.standaloneSetup(new ExampleController())
 .apply(documentationConfiguration(this.restDocumentation))
 .build()
 }
  19. Setup & GET class ExampleControllerSpec extends Specification {
 
 @Rule


    RestDocumentation restDocumentation = new RestDocumentation('src/docs/generated- snippets')
 
 protected MockMvc mockMvc
 
 void setup() {
 this.mockMvc = MockMvcBuilders.standaloneSetup(new ExampleController())
 .apply(documentationConfiguration(this.restDocumentation))
 .build()
 } void 'test and document get with example endpoint'() {
 when:
 ResultActions result = this.mockMvc.perform(get('/hello')
 .contentType(MediaType.APPLICATION_JSON))
  20. Setup & GET class ExampleControllerSpec extends Specification {
 
 @Rule


    RestDocumentation restDocumentation = new RestDocumentation('src/docs/generated- snippets')
 
 protected MockMvc mockMvc
 
 void setup() {
 this.mockMvc = MockMvcBuilders.standaloneSetup(new ExampleController())
 .apply(documentationConfiguration(this.restDocumentation))
 .build()
 } void 'test and document get with example endpoint'() {
 when:
 ResultActions result = this.mockMvc.perform(get('/hello')
 .contentType(MediaType.APPLICATION_JSON)) then:
 result
 .andExpect(status().isOk()) .andExpect(jsonPath('name').value('World'))
 .andExpect(jsonPath('message').value('Hello, World!'))
 .andDo(document('hello-get-example', preprocessResponse(prettyPrint()),
 responseFields(
 fieldWithPath('name').type(JsonFieldType.STRING).description('name'),
 fieldWithPath('message').type(JsonFieldType.STRING).description('hello world'))
 ))
 }
 }
  21. POST void 'test and document post with example endpoint and

    custom name'() {
 when: ResultActions result = this.mockMvc.perform(post(‘/hello') .content(new ObjectMapper() .writeValueAsString(new Example(name: 'mockmvc test’)) .contentType(MediaType.APPLICATION_JSON))
  22. POST void 'test and document post with example endpoint and

    custom name'() {
 when: ResultActions result = this.mockMvc.perform(post(‘/hello') .content(new ObjectMapper() .writeValueAsString(new Example(name: 'mockmvc test’)) .contentType(MediaType.APPLICATION_JSON)) then:
 result
 .andExpect(status().isOk())
 .andExpect(jsonPath('name').value('mockmvc test'))
 .andExpect(jsonPath('message').value('Hello, mockmvc test!')) .andDo(document('hello-post-example', preprocessResponse(prettyPrint()),
 responseFields(
 fieldWithPath('name').type(JsonFieldType.STRING)
 .description('name'),
 fieldWithPath('message').type(JsonFieldType.STRING)
 .description('hello world'))
 ))
 }
  23. List Example void 'test and document get of a list

    from example endpoint'() {
 when:
 ResultActions result = this.mockMvc.perform(get('/hello/list')
 .contentType(MediaType.APPLICATION_JSON))
 
 then:
 result
 .andExpect(status().isOk())
 .andDo(document('hello-list-example', preprocessResponse(prettyPrint()),
 responseFields(
 fieldWithPath('[].message').type(JsonFieldType.STRING)
 .description('message'))
 ))
 }
  24. List Example void 'test and document get of a list

    from example endpoint'() {
 when:
 ResultActions result = this.mockMvc.perform(get('/hello/list')
 .contentType(MediaType.APPLICATION_JSON))
 
 then:
 result
 .andExpect(status().isOk())
 .andDo(document('hello-list-example', preprocessResponse(prettyPrint()),
 responseFields(
 fieldWithPath('[].message').type(JsonFieldType.STRING)
 .description('message'))
 ))
 }
  25. Central Info - Errors void 'test and document error format’()

    { when:
 ResultActions result = this.mockMvc.perform(put('/error')
 .contentType(MediaType.APPLICATION_JSON)
 .requestAttr(RequestDispatcher.ERROR_STATUS_CODE, 405)
 .requestAttr(RequestDispatcher.ERROR_REQUEST_URI, '/hello')
 .requestAttr(RequestDispatcher.ERROR_MESSAGE, "Request method 'PUT' not supported"))

  26. Central Info - Errors void 'test and document error format’()

    { when:
 ResultActions result = this.mockMvc.perform(put('/error')
 .contentType(MediaType.APPLICATION_JSON)
 .requestAttr(RequestDispatcher.ERROR_STATUS_CODE, 405)
 .requestAttr(RequestDispatcher.ERROR_REQUEST_URI, '/hello')
 .requestAttr(RequestDispatcher.ERROR_MESSAGE, "Request method 'PUT' not supported"))
 then:
 result .andExpect(status().isMethodNotAllowed())
 .andDo(document('error-example', preprocessResponse(prettyPrint()),
 responseFields(
 fieldWithPath(‘error') .description('The HTTP error that occurred, e.g. `Bad Request`'),
 fieldWithPath(‘message') .description('A description of the cause of the error'),
 fieldWithPath('path').description('The path to which the request was made'),
 fieldWithPath('status').description('The HTTP status code, e.g. `400`'),
 fieldWithPath(‘timestamp') .description('The time, in milliseconds, at which the error occurred'))
 ))
 }
  27. Add Snippets to AsciiDoc == Errors
 
 Whenever an error

    response (status code >= 400) is returned, the body will contain a JSON object
 that describes the problem. The error object has the following structure:
 
 include::{snippets}/error-example/response-fields.adoc[]
 
 For example, a request that attempts to apply a non-existent tag to a note will produce a
 `400 Bad Request` response:
 
 include::{snippets}/error-example/http-response.adoc[]
 
 [[resources]]
 = Resources
 
 include::resources/example.adoc[]
  28. Publish Docs publish.gradle buildscript {
 repositories {
 jcenter()
 }
 


    dependencies {
 classpath 'org.ajoberstar:gradle-git:1.1.0'
 }
 }
 
 apply plugin: 'org.ajoberstar.github-pages'
 
 githubPages {
 repoUri = '[email protected]:jlstrater/groovy-spring-boot-restdocs-example.git'
 pages {
 from(file('build/docs'))
 }
 }
  29. Conclusion • API documentation is complex • Choosing the right

    tool for the job not just about the easiest one to setup • Spring REST Docs is a promising tool to enforce good testing and documentation practices without muddying source code.