Writing OpenAPI (Swagger) Specification Tutorial Series - Part 5

Advanced Input And Output Modeling

By Arnaud Lauret, May 6, 2016

After learning how to create an accurate data model, we continue to delve into the OpenAPI specification’s and discover how to describe tailor made API’s inputs and outputs.

Writing OpenAPI (Swagger) Specification Tutorial Series

This tutorial teaches everything about the OpenAPI 2.0 Specification (fka. as Swagger), most of what you’ll read here can still be applied on version 3.

If you’re a bit lost in the specification (version 2 or 3), take a look at the OpenAPI Map:

In previous parts (especially The basics and Simplifying specification file we have learned how to describe simple operations parameters and responses using inline definitions or high level ones. In this fifth part you will discover all the tips and tricks to describe highly accurate parameters and responses.

Parameters

In this section you will learn to define:

  • Required or optional parameter
  • Parameter with default value
  • Parameter with empty value
  • Array parameter
  • Header parameter
  • Form parameter
  • File parameter
  • Parameter’s media type

Required or optional parameter

We already have used the required key word which is used to define a mandatory parameter or a mandatory value in definition.

Defining a required or optional parameter

In a parameter, required is an optional value which type is boolean. Its default value is false.

When used in a operation, the username parameter is mandatory:

  username:
    name: username
    in: path
    required: true
    description: The person's username
    type: string

When used in an operation, the pageSize parameter is NOT mandatory (required is not defined and therefore is false).

  pageSize:
    name: pageSize
    in: query
    description: Number of persons returned
    type: integer
    format: int32
    minimum: 0
    exclusiveMinimum: true
    maximum: 100
    exclusiveMaximum: false
    multipleOf: 10
    default: 20

Defining a required or optional property in a definition used as a parameter

In a definition, required is an optional value which type is a list of string. This list contains the mandatory properties names. A property which is not referenced in this list is NOT mandatory. If required is not defined, all object properties are not mandatory. When this definition is used on a request, all required properties MUST be provided.

In the definition Person which is used as a body parameter in POST /persons, the username property is mandatory (present in required list) , all others are not mandatory (not referenced in required list).

  Person:
    required:
      - username
    properties:
      firstName:
        type: string
      lastName:
        type: string
      username:
        type: string
        pattern: '[a-z0-9]{8,64}'
        minLength: 8
        maxLength: 64
      dateOfBirth:
        type: string
        format: date
      lastTimeOnline:
        type: string
        format: date-time
        readOnly: true
      avatarBase64PNG:
        type: string
        format: byte
        default: 
      spokenLanguages:
        $ref: '#/definitions/SpokenLanguages'

Parameter with default value

By using the keyword default you can define a default value for a parameter or a default value for a property in a definition. This default value is the one that the server will use if none is provided. Using default does not make sense when a property or parameter is required.

Defining a parameter’s default value

On the parameter pageSize, we set its default value to 20. If this value is not provided, the server will then return pages containing 20 elements.

  pageSize:
    name: pageSize
    in: query
    description: Number of persons returned
    type: integer
    format: int32
    minimum: 0
    exclusiveMinimum: true
    maximum: 100
    exclusiveMaximum: false
    multipleOf: 10
    default: 20

On the parameter pageNumber, we set its default value to 1. If this value is not provided, the server will then return the first page.

  pageNumber:
    name: pageNumber
    in: query
    description: Page number
    type: integer
    default: 1

Defining a property’s default value in a definition used as a parameter

On the definition Person, the avatarBase64PNG property default value is a 64x64 pixels API Handyman PNG icon as a base64 string.

Default Avatar

  Person:
    required:
      - username
    properties:
      firstName:
        type: string
      lastName:
        type: string
      username:
        type: string
        pattern: '[a-z0-9]{8,64}'
        minLength: 8
        maxLength: 64
      dateOfBirth:
        type: string
        format: date
      lastTimeOnline:
        type: string
        format: date-time
        readOnly: true
      avatarBase64PNG:
        type: string
        format: byte
        default: 
      spokenLanguages:
        $ref: '#/definitions/SpokenLanguages'

Defining a default value in a hashmap definition used as a parameter

In previous post (Advanced data modeling) we’ve learned how to define hashmaps. We can set a default value in a an hashmap this way:

  SpokenLanguages:
    additionalProperties:
      type: string
    properties:
      defaultLanguage:
        type: string
        default: english

Just like we’ve learned on previous post, we define a property defaultLanguage on SpokenLanguages definition and today we set a default value for this property. Therefore is the property spokenLanguages of the Person definition (which is a SpokenLanguages) is not provided on POST /persons, its value will be {"defaultLanguage": "english"}

Parameter with empty value

If we want to add a filter parameter to include non verified users on GET /persons a first idea would be to have something like GET /persons?page=2&includeVerifiedUsers=true. But why should we have to set a value when the parameters name is sufficient to express what we want to do? Having GET /persons?page=2&includeVerifiedUsers would be much better.

To do that we just need to use the allowEmptyValue key word on the parameter’s description. We have also define a default value to false, therefore GET /persons?page=2&includeVerifiedUsers will include verified users and GET /persons?page=2 will not.

  includeNonVerifiedUsers:
    name: includeNonVerifiedUsers
    in: query
    type: boolean
    default: false
    allowEmptyValue: true

Array parameter

When it comes to sorting or filtering an API designer almost always ends asking himself: but how will I define an array parameter on a get request in my OpenAPI specification?

It’s easy, he just need to define an array type attribute and use the accurate collectionFormat:

collectionFormat   Description
csv (default value) Comma separated values foo,bar
ssv Space separated values foo bar
tsv Tab separated values foo\tbar
pipes Pipes separated values foo|bar
multi Corresponds to multiple parameter instances instead of multiple values for a single instance foo=bar&foo=baz. This is valid only for parameters in query or formData.

Defining a separated values parameter

If we want to sort a persons list on multiple parameters (username, firstname, lastname, lastTimeOnline) ascending or descending we could use something like GET /persons?sort=-lastTimeOnline|+firtname|+lastname. The sort parameters are in a pipe separated array named sort, each of them starting with + for an ascending sort or - for a descending one.

This sortparameter is defined with an array of string using a pipes collectionFormat:

  sortPersons:
    name: sort
    in: query
    type: array
    uniqueItems: true
    minItems: 1
    maxItems: 3
    collectionFormat: pipes
    items:
      type: string

We can now have requests like this: GET /persons?sort=String1|String2|String3. But how do we set the sorting direction and how can we we enforce the values within the array?

In this specific use case we will define a pattern for the string within the array, just like we’ve learned on previous part when defining the username property. This pattern says that a value within the array may start with + (for ascending) and - (for descending) and be followed by username, firstname, lastname or lastTimeOnline.

  sortPersons:
    name: sort
    in: query
    type: array
    uniqueItems: true
    minItems: 1
    maxItems: 3
    collectionFormat: pipes
    items:
      type: string
      pattern: '[-+](username|lastTimeOnline|firstname|lastname)'

Our API is now ready to handle a GET /persons?sort=-lastTimeOnline|+firtname|+lastname request.

And icing on the cake, we can also define a default sort (last online time descending and username ascending) by adding a default value on the array level:

  sortPersons:
    name: sort
    in: query
    type: array
    uniqueItems: true
    minItems: 1
    maxItems: 3
    collectionFormat: pipes
    items:
      type: string
      pattern: '[-+](username|lastTimeOnline|firstname|lastname)'
    default:
      - -lastTimeOnline
      - +username

Defining a multiple values parameter

If we want to filter a persons collected items list on mutiple items types we could use something like GET /persons/apihandyman/collected-items?itemType=AudioCassette&itemType=Vinyl. The filter parameters are in a multi array named itemType, each value corresponding to one of the item’s type we want to filter on.

This filterItemTypes parameter is defined with an array of unique stringusing a multi colectionFormat:

  filterItemTypes:
    name: itemType
    in: query
    type: array
    collectionFormat: multi
    uniqueItems: true
    items:
      type: string

And icing on the cake we enforce the possible values by defining an enum on the string within the array:

  filterItemTypes:
    name: itemType
    in: query
    type: array
    collectionFormat: multi
    uniqueItems: true
    items:
      type: string
      enum:
        - AudioCassette
        - Vinyl
        - VHS

Parameter location is not only path, query or body

When describing a parameter, the keyword in is used to set its location. We already have seen in previous parts the most common values of in are:

  • path
  • query
  • body

But they are not the only ones, there two others:

  • header
  • formData

We’ll see in next sections how we can use them in an API definition to define

  • Header parameter
  • Form parameter
  • File parameter

Header parameter

Let’s say we want our API consumer’s to provide some informations about themselves by using the good old User-Agent HTTP header (for tracking, debugging, or whatever you want).

We define the parameter just like any other one, we just need to set the headervalue in in:

  userAgent:
    name: User-Agent
    type: string
    in: header
    required: true

And we use it (on every path) just like any other parameter:

paths:
  /persons:
    parameters:
      - $ref: '#/parameters/userAgent'

There’s absolutely no way of telling once and for all that all operations needs this header parameter (for now).

nb: This works also with custom HTTP header.

Form parameter

Let’s say we have a partner who need to use our API in a js-less-browser environment to create persons. He can only provide the creation information in a good old HTML form format:

POST /js-less-persons

username=apihandyman&firstname=API&lastname=Handyman

No problem, we just need to define all form parameters with in sets to formData and setting the consumes media type to application/x-www-form-urlencoded:

    post:
      summary: Creates a person
      description: For JS-less partners
      consumes:
        - application/x-www-form-urlencoded
      produces:
        - text/html
      parameters:
        - name: username
          in: formData
          required: true
          pattern: '[a-z0-9]{8,64}'
          minLength: 8
          maxLength: 64
          type: string
        - name: firstname
          in: formData
          type: string
        - name: lastname
          in: formData
          type: string
        - name: dateOfBirth
          in: formData
          type: string
          format: date
      responses:
        '204':
          description: Person succesfully created.

File parameter

To define an operation which will accept a file as input parameter, you need to:

  • use multipart/form-data media type
  • set parameter’s in value to formData
  • set parameter’s type value to file

  /images:
    parameters:
      - $ref: '#/parameters/userAgent'
    post:
      summary: Uploads an image
      consumes:
        - multipart/form-data
      parameters:
        - name: image
          in: formData
          type: file
      responses:
        '200':
          description: Image's ID
          schema:
            properties:
              imageId:
                type: string

And what if I want to restrict file’s media type? Unfortunately, it’s not possible for now:

The spec doesn’t allow specifying a content type for specific form data parameters. It’s a limitation of the spec. Ron Ratovsky comment in Swagger UI 609 issue

Parameter’s media types

An API can consume various media types, the most common one is application/json, but it’s not the only one an API can use. The keyword consumes on root or operation level is used to describe the accepted media types list.

We set the global media-type on the root level of the OpenAPI document, here our API consumes JSON and YAML:

consumes:
  - application/json
  - application/x-yaml

This settings can be overriden on the operation level by redefining the consumes list:

  /images:
    parameters:
      - $ref: '#/parameters/userAgent'
    post:
      summary: Uploads an image
      consumes:
        - multipart/form-data
      parameters:
        - name: image
          in: formData
          type: file
      responses:
        '200':
          description: Image's ID
          schema:
            properties:
              imageId:
                type: string

Responses

In this section you will learn to define:

  • Response without a body
  • Required or optional values in response
  • Response’s headers
  • Default response
  • Response’s media types

Response without a body

It’s a common feature in HTTP protocole and in REST API to have response without body. For example the 204 HTTP Status is used by a server which wants to indicate a succes without returning any content (or body).

To define a response without a body, all you have to do is set its status and description:

    post:
      summary: Creates a person
      description: Adds a new person to the persons list.
      parameters:
        - name: person
          in: body
          required: true
          description: The person to create.
          schema:
            $ref: '#/definitions/Person'
      responses:
        '204':
          description: Person succesfully created.

Required or optional values in response

Just like we’ve seen on parameters, we can define mandatory properties in definition used in responses. Mandatory properties are defined with the required list. A property referenced in this list MUST be sent by the server and a property absent from this list MAY be sent by the server.

When GET /persons/apihandyman returns a Person, this person will surely contains a username, but all other values may not be sent:

  Person:
    required:
      - username
    properties:
      firstName:
        type: string
      lastName:
        type: string
      username:
        type: string
        pattern: '[a-z0-9]{8,64}'
        minLength: 8
        maxLength: 64
      dateOfBirth:
        type: string
        format: date
      lastTimeOnline:
        type: string
        format: date-time
        readOnly: true
      avatarBase64PNG:
        type: string
        format: byte
        default: 
      spokenLanguages:
        $ref: '#/definitions/SpokenLanguages'

Response’s headers

The HTTP status and the body are not the only way to provide information on the result of an API call, HTTP headers can be used too.

Let’s say we want to offer something like the Twitter’s API rate limiting information to provide the remaining number of API calls and when the limit will be reset. We have to define two header X-Rate-Limit-Remaining and X-Rate-Limit-Reset and each API response:

    post:
      summary: Creates a person
      description: Adds a new person to the persons list.
      parameters:
        - name: person
          in: body
          required: true
          description: The person to create.
          schema:
            $ref: '#/definitions/Person'
      responses:
        '204':
          description: Person succesfully created.
          headers:
            X-Rate-Limit-Remaining:
              type: integer
            X-Rate-Limit-Reset:
              type: string
              format: date-time

Unfortunately there’s absolutely no way of defining such headers once and for all (for now).

Default response

The OpenAPI specification allow to define a default response on each operation.

This could be used to define a single generic response but your API definition would not be easily understandable. I’d prefer to use it in conjunction with a full description of which HTTP status an API handle, and then use the default response to say if you have any other HTTP status than the ones I described explicitely, it do not comes from this API.

A generic response defined once and for all in the responses section:

  TotallyUnexpectedResponse:
    description: A totally unexpected response

This response used as a default response on each operation:

    delete:
      summary: Deletes a person
      description: Delete a single person identified via its username
      responses:
        '204':
          description: Person successfully deleted.
          headers:
            X-Rate-Limit-Remaining:
              type: integer
            X-Rate-Limit-Reset:
              type: string
              format: date-time
        '404':
          $ref: '#/responses/PersonDoesNotExistResponse'
        '500':
          $ref: '#/responses/Standard500ErrorResponse'
        default:
          $ref: '#/responses/TotallyUnexpectedResponse'

Unfortunately there’s absolutely no way of defining a global response once and for all (for now).

Response’s media types

An API can produce various media types, the most common one is application/json, but it’s not the only one an API can use. The keyword produces on root or operation level is used to describe the returned media types list.

We set the global media-type on the root level of the OpenAPI document, here our API produces JSON and YAML:

produces:
  - application/json
  - application/x-yaml

This settings can be overriden on the operation level by redefining the local produces list, here’s an example on GET /images/{imageId} operation which return an image:

  /images/{imageId}:
    parameters:
      - $ref: '#/parameters/userAgent'
    get:
      summary: Gets an image
      parameters:
        - name: imageId
          in: path
          required: true
          type: string
      produces:
        - image/png
        - image/gif
        - image/jpeg
        - application/json
        - application/x-yaml
      responses:
        '200':
          description: The image
          headers:
            X-Rate-Limit-Remaining:
              type: integer
            X-Rate-Limit-Reset:
              type: string
              format: date-time
        '404':
          description: Image do not exists
          headers:
            X-Rate-Limit-Remaining:
              type: integer
            X-Rate-Limit-Reset:
              type: string
              format: date-time
        '500':
          $ref: '#/responses/Standard500ErrorResponse'
        default:
          $ref: '#/responses/TotallyUnexpectedResponse'

Note that the produces contains images media types but also json and yaml media types as if there’s a 500 error, the message will be return with one of those 2 media types.

How to use a single definition when output returns more data than input needs

And finally, as already seen on previous part in section 2.1, in a definition the readOnly property is set to true to indicate that this property MAY be sent in a response and MUST NOT be sent in a request.

It allows you to use a single definition when output returns more data than input needs.

In the definition Person which is both used as a part of the response in GET /persons and a body parameter in POST /persons the lastTimeOnline property has readOnly set to true. Therefore this property MUST NOT be provided on POST /persons and MAY be returned by the server on GET /persons:

  Person:
    required:
      - username
    properties:
      firstName:
        type: string
      lastName:
        type: string
      username:
        type: string
        pattern: '[a-z0-9]{8,64}'
        minLength: 8
        maxLength: 64
      dateOfBirth:
        type: string
        format: date
      lastTimeOnline:
        type: string
        format: date-time
        readOnly: true
      avatarBase64PNG:
        type: string
        format: byte
        default: 
      spokenLanguages:
        $ref: '#/definitions/SpokenLanguages'

Warning: SwaggerUI does not handle this yet (issue 884).

Conclusion

You are now a Jedi of API’s input and ouput definition with the OpenAPI specification. In next post you’ll learn to describe how your API is secured.

By continuing to use this web site you agree with the API Handyman website privacy policy (effective date , June 28, 2020).