Writing OpenAPI (Swagger) Specification Tutorial Series - Part 8
Splitting specification file
By Arnaud Lauret, August 2, 2016
With previous posts we have learned to produce an OpenAPI specification containing all OpenAPI specification subtleties. Some specification files may become quite large or may contain elements which could be reused in other APIs. Splitting a specification file will help to keep it maintainable by creating smaller files and also help to ensure consistency throughout APIs by sharing common elements.
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 we’ve learned to create highly accurate API description which can become quite large or may contain elements that can be reused, in this eighth part we’ll learn how to split an OpenAPI specification file into smaller and reusable elements.
JSON Pointers
In part 3 - Simplifying spefication file we have learned how to simplify the specification by creating reusable elements. In the example below, the Person
definition is defined once in definitions
and used as
- A body parameter in
POST /persons
- A response schema in
GET /persons/{username}
- A sub-elements in
Persons
definition
swagger: "2.0"
info:
version: 1.0.0
title: Simple API
description: A simple API to learn how to write OpenAPI Specification
schemes:
- https
host: simple.api
basePath: /openapi101
paths:
/persons:
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
parameters:
- name: pageSize
in: query
description: Number of persons returned
type: integer
- name: pageNumber
in: query
description: Page number
type: integer
responses:
200:
description: A list of Person
schema:
$ref: "#/definitions/Persons"
post:
summary: Creates a person
description: Adds a new person to the persons list.
parameters:
- name: person
in: body
description: The person to create.
schema:
$ref: "#/definitions/Person"
responses:
204:
description: Persons succesfully created.
400:
description: Persons couldn't have been created.
/persons/{username}:
get:
summary: Gets a person
description: Returns a single person for its username.
parameters:
- name: username
in: path
required: true
description: The person's username
type: string
responses:
200:
description: A Person
schema:
$ref: "#/definitions/Person"
404:
description: The Person does not exists.
definitions:
Person:
required:
- username
properties:
firstName:
type: string
lastName:
type: string
username:
type: string
Persons:
type: array
items:
$ref: "#/definitions/Person"
To use the Person
definition in these different places, we use a JSON Pointer (defined by RFC6901):
This pointer describes a path in the document, pointing to Person
in definitions
which is as the root (#
) of the current document.
But JSON pointers are not only meant to point something within the current document, they can be used to reference something in another document.
Basic splitting
Let’s see how we can split the file we created in part 3
Referencing a local file
We can create a person.yaml
file containing the Person
definition:
Person:
required:
- username
properties:
firstName:
type: string
lastName:
type: string
username:
type: string
Then we can remove the Person
definition in definitions
and replace all existing references to person #definitions/Person
by a reference to Person
in the person.yaml
file:
swagger: "2.0"
info:
version: 1.0.0
title: Simple API
description: A simple API to learn how to write OpenAPI Specification
schemes:
- https
host: simple.api
basePath: /openapi101
paths:
/persons:
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
parameters:
- name: pageSize
in: query
description: Number of persons returned
type: integer
- name: pageNumber
in: query
description: Page number
type: integer
responses:
200:
description: A list of Person
schema:
$ref: "#/definitions/Persons"
post:
summary: Creates a person
description: Adds a new person to the persons list.
parameters:
- name: person
in: body
description: The person to create.
schema:
$ref: "person.yaml#/Person"
responses:
204:
description: Persons succesfully created.
400:
description: Persons couldn't have been created.
/persons/{username}:
get:
summary: Gets a person
description: Returns a single person for its username.
parameters:
- name: username
in: path
required: true
description: The person's username
type: string
responses:
200:
description: A Person
schema:
$ref: "person.yaml#/Person"
404:
description: The Person does not exists.
definitions:
Persons:
type: array
items:
$ref: "person.yaml#/Person"
Editing splitted local files with the online editor
Tools will look for the referenced file (person.yaml
) in the same directory as the file containing the reference.
But when you use the online editor it does not make sense, such local reference cannot be resolved.
Fortunately the editor propose a configuration allowing to set a server to resolve these local references. This configuration is accessible in the Preferences->Preferences menu:
The Pointer Resolution Base Path configuration is on the bottom of the configuration screen:
All you need to do is put the referenced yaml files into a web server (with CORS activated) and modify the editor’s configuration to point this server.
http-server a lightweight web server
You can use http-server a lightweight node.js web server:
You now can start a web service on any folder:
or
By default, http-server
listens on 8080 port, files will be accessble through http://localhost:8080/<path to file within folder>
.
The --cors
flag is used to activate CORS directive and allow XHR request to this local webserver from the online editor page.
Without CORS activated, the editor will not be able to download files.
Modifying online editor configuration
Once the web server is started you can modifiy the editor to set the URL to http://localhost:8080/
:
Files referenced with $ref: <filename>#/example
will be downloaded from http://localhost:8080/<filename>
.
Once this is done, you may need to do a force refresh (clean cache) of the editor’s the page to get the green bar.
Cache
When you add a reference to a new file (that was not already reference), this new file may not be downloaded automatically resulting in errors (Reference could not be resolved: newfile.yaml
). You may need to refresh the editor’s page with cache cleaning to solve this error.
Folders
You’re under no obligation to put all sub-files on a “root” level, you can store them in differents sub-folders.
Reference to a file in a folder
Let’s move the person.yaml
file into a sub-folder folder
, the new reference will be like this:
File referencing a file from an upper folder
You can also reference a file outside the current folder. Let’s create a persons.yaml
file into a another-folder
folder:
This file reference the person.yaml
file using ..
to get to the upper level.
To use persons.yaml
, we proceed just like with person.yaml
. We remove the Persons
definition from the main file and we replace its reference #/definitions/Persons
by another-folder/persons.yaml#Persons
.
Here’s the full file with all definitions externalized (the definitions
section has been removed):
swagger: "2.0"
info:
version: 1.0.0
title: Simple API
description: A simple API to learn how to write OpenAPI Specification
schemes:
- https
host: simple.api
basePath: /openapi101
paths:
/persons:
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
parameters:
- name: pageSize
in: query
description: Number of persons returned
type: integer
- name: pageNumber
in: query
description: Page number
type: integer
responses:
200:
description: A list of Person
schema:
$ref: "another-folder/persons.yaml#/Persons"
post:
summary: Creates a person
description: Adds a new person to the persons list.
parameters:
- name: person
in: body
description: The person to create.
schema:
$ref: "folder/person.yaml#/Person"
responses:
204:
description: Persons succesfully created.
400:
description: Persons couldn't have been created.
/persons/{username}:
get:
summary: Gets a person
description: Returns a single person for its username.
parameters:
- name: username
in: path
required: true
description: The person's username
type: string
responses:
200:
description: A Person
schema:
$ref: "folder/person.yaml#/Person"
404:
description: The Person does not exists.
Referencing a remote files
As you may have guess while modifying the references in the specification and editor configuration, it is also possible to reference a remote file.
Remote files
All we need to do is to put the full file’s URL in the reference:
Remember that the server MUST have CORS activated to allow the editor to download the file.
We have launched a web server on 8080 port, so all we have to do is add http://localhost:8080/
to the references to Person
:
Remote files containing local references
What happen if the remote file reference a local file? (Main file -> Remote file -> Local file
).
The parser will seek this “local” file on the remote server providing the remote file.
If we replace the local to Persons
by a remote reference…
… the person.yaml
file referenced “locally” in the persons.yaml
file will be loaded from http://localhost:8080
which has served the persons.yaml
.
Remote files containing remote references
If a remote file contains a remote reference (Main file -> Remote file -> Remote file
), it will be resolved like in 2.4.1 Remote files.
We can replace the person.yaml
file local reference by a remote reference in persons.yaml
:
Multiple items in a single sub-file
We have put Person
and Persons
definitions in separate files: this is not an obligation. You can put more than one item in a single file, you only need to use the right JSON Pointer.
Person and Person in a single file
If we concatenate person.yaml
and persons.yaml
into a single file called definitions.yaml
:
Person:
required:
- username
properties:
firstName:
type: string
lastName:
type: string
username:
type: string
Persons:
type: array
items:
$ref: "#/Person"
Note that in Persons
the reference to Person
is now #/Person
.
In the main file we only need to change the filename when reference these two definitions:
Organizing content in a sub-file
Within the sub-file you can organize the content as you wish. Here Person
is in SomeDefinitions
and Persons is in OtherDefinitions
:
SomeDefinitions:
Person:
required:
- username
properties:
firstName:
type: string
lastName:
type: string
username:
type: string
OtherDefinitions:
Persons:
type: array
items:
$ref: "#/SomeDefinitions/Person"
Note that in Persons
the reference to Person
is now #/SomeDefinitions/Person
.
In the main file we have to modify the path to get the item in the sub-file for these two definitions:
Definitions, Responses, Parameters
What we have done with Person
and Persons
definitions can be done with any reusable items (i.e. using $ref
) such as responses and parameters in the Open API Specification file.
OpenAPI chainsaw massacre
Thanks to Mohsen Azimi’s post I’ve discovered that using sub-files for definitions, responses and parameters which obviously use $ref JSON Pointers is not the only way of using sub-files. You can use sub-files for almost anything in the specification.
We will split the huge file created in previous post about documentation.
Let the chainsaw massacre begin
Let’s start with the info
section:
swagger: '2.0'
info:
version: 1.1.0
title: Simple API
description: |
A simple API to learn how to write OpenAPI Specification.
This file uses almost every single aspect of the [Open API Specification](https://openapis.org/).
This API will use JSON.
JSON looks like this:
```JSON
{
"key": "value",
"anotherKey": "anotherValue"
}
```
termsOfService: http://simple.api/terms-of-service
contact:
name: John Doe
url: http://simple.api/contact
email: [email protected]
license:
name: Apache-2.0
url: http://www.apache.org/licenses/LICENSE-2.0
We can put its whole content in a file called info.yaml
:
version: 1.1.0
title: Simple API
description: |
A simple API to learn how to write OpenAPI Specification.
This file uses almost every single aspect of the [Open API Specification](https://openapis.org/).
This API will use JSON.
JSON looks like this:
```JSON
{
"key": "value",
"anotherKey": "anotherValue"
}
```
termsOfService: http://simple.api/terms-of-service
contact:
name: John Doe
url: http://simple.api/contact
email: [email protected]
license:
name: Apache-2.0
url: http://www.apache.org/licenses/LICENSE-2.0
And reference it just like this in info
:
The wizard of resolution
We can do the same thing with paths
, definitions
, responses
and parameters
: copy the section content in a sub-file (paths.yaml, definitions.yaml, responses.yaml and parameters.yaml) and reference it in the main file:
paths:
$ref: paths.yaml
definitions:
$ref: definitions.yaml
responses:
$ref: responses.yaml
parameters:
$ref: parameters.yaml
If you remember previous posts, these sections references each other in many ways, for example in paths.yaml
:
The username
parameter is no longer in the main file and is not in the paths.yaml
file but in the parameters.yaml
file:
How could the main file be considered valid in the editor (or in any tool parsing the file)? It’s simply because the sub-files are loaded and then the whole content (file + sub-files) is validated.
A less rough split
We can use JSON pointers for almost anything in the specification as as long as the $ref JSON pointer reference something corresponding to the expected object (or value) in the OpenAPI specification.
Referencing object in custom structure
And as seen earlier in this post we can put many items in a sub-file.
The documentation.yaml
file contains the data for externalDocs
and tags
(note the custom structure on line 1 and 6):
external:
description: |
**Complete** documentation describing how to use this API
url: http://doc.simple.api/
categories:
- name: Persons
description: Everything you need to handle `users` and `friends`
externalDocs:
description: People category documentation
url: http://doc.simple.api/people
- name: Items
description: Everything you need to handle items collected by users
externalDocs:
description: Items category documentation
url: http://doc.simple.api/items
- name: Media
description: Everything you need to handle images
externalDocs:
description: Media category documentation
url: http://doc.simple.api/media
- name: JSLess
description: Specific operations for JS less consumers
externalDocs:
description: JS Less Consumers documentation
url: http://doc.simple.api/jsless
These data are referenced this way in the main file:
The security.yaml
file contains the data for securityDefinitions
and security
(note the custom structure on line 20):
securityDefinitions:
OauthSecurity:
description: New Oauth security system. Do not use MediaSecurity or LegacySecurity.
type: oauth2
flow: accessCode
authorizationUrl: 'https://oauth.simple.api/authorization'
tokenUrl: 'https://oauth.simple.api/token'
scopes:
admin: Admin scope
user: User scope
MediaSecurity:
description: Specific media security for backward compatibility. Use OauthSecurity instead.
type: apiKey
in: query
name: media-api-key
LegacySecurity:
description: Legacy security system for backward compatibility. Use OauthSecurity instead.
type: basic
defaultSecurity:
- OauthSecurity:
- user
- LegacySecurity: []
Referencing a string or a simple list
It also work for simpler value like a string or a list of string. The schema
, host
and basepath
values can be moved into a single file security.yaml
(note the custom structure on line 3 and 4):
And referenced like this
schemes:
$ref: endpoint.yaml#/schemes
host:
$ref: endpoint.yaml#/server_and_port
basePath:
$ref: endpoint.yaml#/path
Reusing a value
As long as the API consumes and produces the same media types, we define a single value in the mediatypes.yaml
file:
And then we reference this single value in both produces
and consumes
:
A smarter split
The API we built with the previous parts could be divided in four parts:
- Common items like headers, media types, security, definitions, parameters and responses
- Persons operations, parameters, responses and definitions
- Legacy operations
- Images operations
We can create 4 sub-files and reference them from a main file.
commons.yaml
In the commons.yaml
file we put every items that can be reused across other files:
securityDefinitions:
OauthSecurity:
description: New Oauth security system. Do not use MediaSecurity or LegacySecurity.
type: oauth2
flow: accessCode
authorizationUrl: 'https://oauth.simple.api/authorization'
tokenUrl: 'https://oauth.simple.api/token'
scopes:
admin: Admin scope
user: User scope
MediaSecurity:
description: Specific media security for backward compatibility. Use OauthSecurity instead.
type: apiKey
in: query
name: media-api-key
LegacySecurity:
description: Legacy security system for backward compatibility. Use OauthSecurity instead.
type: basic
defaultSecurity:
- OauthSecurity:
- user
- LegacySecurity: []
defaultMediatypes:
- application/json
- application/x-yaml
defaultHeaders:
X-Rate-Limit-Remaining:
description: How many calls consumer can do
type: integer
X-Rate-Limit-Reset:
description: When rate limit will be reset
type: string
format: date-time
parameters:
userAgent:
name: User-Agent
description: All API consumers MUST provide a user agent
type: string
in: header
required: true
pageSize:
name: pageSize
in: query
description: Number of items returned
type: integer
format: int32
minimum: 0
exclusiveMinimum: true
maximum: 100
exclusiveMaximum: false
multipleOf: 10
default: 20
pageNumber:
name: pageNumber
in: query
description: Page number
type: integer
default: 1
responses:
Standard500ErrorResponse:
description: An unexpected error occured.
headers:
$ref: '#/defaultHeaders'
schema:
$ref: '#/definitions/Error'
TotallyUnexpectedResponse:
description: A totally unexpected response
headers:
$ref: '#/defaultHeaders'
definitions:
ErrorMessage:
title: MultiDeviceErrorMessage
description: An error message with a long and a short description
required:
- longMessage
- shortMessage
properties:
longMessage:
description: A long error description
type: string
shortMessage:
description: A short error description
type: string
MultilingualErrorMessage:
title: MultiLingualMultiDeviceErrorMessage
description: An multilingual error message (hashmap) with a long and a short description
additionalProperties:
$ref: '#/definitions/ErrorMessage'
properties:
defaultLanguage:
$ref: '#/definitions/ErrorMessage'
example:
defaultLanguage:
longMessage: We're deeply sorry but an error occured
shortMessage: Error
fr:
longMessage: Nous sommes désolé mais une erreur est survenu
shortMessage: Erreur
Error:
title: MultiLingualMultiDeviceError
description: Give full information about the problem
required:
- code
- message
properties:
code:
description: A human readable code (death to numeric error codes!)
type: string
enum:
- DBERR
- NTERR
- UNERR
example: UNERR
message:
$ref: '#/definitions/MultilingualErrorMessage'
Paging:
required:
- totalItems
- totalPages
- pageSize
- currentPage
properties:
totalItems:
type: integer
totalPages:
type: integer
pageSize:
type: integer
currentPage:
type: integer
images.yaml
In the images.yaml
file we put both /images
and /images/{id}
paths informations:
images:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
post:
summary: Uploads an image
description: Upload an image, will return an image id.
operationId: storeImage
externalDocs:
description: How to upload media
url: http://doc.simple.api/media/upload
tags:
- Media
security:
- MediaSecurity: []
consumes:
- multipart/form-data
parameters:
- name: image
in: formData
type: file
responses:
'200':
description: Image's ID
schema:
properties:
imageId:
type: string
headers:
$ref: commons.yaml#/defaultHeaders
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
images-imageId:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets an image
description: Return an image
operationId: readImage
tags:
- Media
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:
$ref: commons.yaml#/defaultHeaders
'404':
description: Image do not exists
headers:
$ref: commons.yaml#/defaultHeaders
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
Note that all references to common items point to commons.yaml
.
legacy.yaml
Same for the /js-less-consumer-persons
path we put in legacy.yaml
file in js-less-consumer-persons
:
js-less-consumer-persons:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
post:
summary: Creates a person
description: For JS-less partners
operationId: createUserJS
deprecated: true
tags:
- JSLess
- Persons
security:
- OauthSecurity:
- admin
- LegacySecurity: []
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.
headers:
$ref: 'commons.yaml#/defaultHeaders'
'400':
description: Person couldn't have been created.
headers:
$ref: 'commons.yaml#/defaultHeaders'
'500':
description: An error occured.
headers:
$ref: 'commons.yaml#/defaultHeaders'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
persons.yaml
All persons items go in the persons.yaml
:
- each path go in its own named section (like
persons-username
for/persons/{username}
) - all persons specific definitions goes in
definitions
(and can be referenced with#definition/name
) - all persons specific responses goes in
responses
(and can be referenced with#responses/name
) - all persons specific parameters goes in
parameters
(and can be referenced with#parameters/name
)
persons:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
operationId: searchUsers
tags:
- Persons
parameters:
- $ref: 'commons.yaml#/parameters/pageSize'
- $ref: 'commons.yaml#/parameters/pageNumber'
- $ref: '#/parameters/includeNonVerifiedUsers'
- $ref: '#/parameters/sortPersons'
responses:
'200':
description: A list of Person
schema:
$ref: '#/definitions/Persons'
headers:
$ref: commons.yaml#/defaultHeaders
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
post:
summary: Creates a person
description: Adds a new person to the persons list.
operationId: createUser
tags:
- Persons
security:
- OauthSecurity:
- admin
- LegacySecurity: []
parameters:
- name: person
in: body
required: true
description: The person to create.
schema:
$ref: '#/definitions/Person'
responses:
'204':
description: Person succesfully created.
headers:
$ref: commons.yaml#/defaultHeaders
'400':
description: Person couldn't have been created.
headers:
$ref: commons.yaml#/defaultHeaders
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
persons-username:
parameters:
- $ref: '#/parameters/username'
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets a person
description: Returns a single person for its username.
operationId: readPerson
tags:
- Persons
responses:
'200':
description: A Person
schema:
$ref: '#/definitions/Person'
headers:
$ref: commons.yaml#/defaultHeaders
'404':
$ref: '#/responses/PersonDoesNotExistResponse'
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
delete:
summary: Deletes a person
description: Delete a single person identified via its username
operationId: deletePerson
tags:
- Persons
responses:
'204':
description: Person successfully deleted.
headers:
$ref: commons.yaml#/defaultHeaders
'404':
$ref: '#/responses/PersonDoesNotExistResponse'
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
persons-username-friends:
parameters:
- $ref: '#/parameters/username'
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets a person's friends
description: Returns a list containing all persons. The list supports paging.
operationId: readPersonsFriends
tags:
- Persons
parameters:
- $ref: 'commons.yaml#/parameters/pageSize'
- $ref: 'commons.yaml#/parameters/pageNumber'
- $ref: '#/parameters/includeNonVerifiedUsers'
- $ref: '#/parameters/sortPersons'
responses:
'200':
description: A person's friends list
schema:
$ref: '#/definitions/PagedPersons'
headers:
$ref: commons.yaml#/defaultHeaders
'404':
$ref: '#/responses/PersonDoesNotExistResponse'
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
persons-username-collecting-items:
parameters:
- $ref: '#/parameters/username'
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets a person's collecting items list
description: |
Returns a list containing all items this person is looking for.
The list supports paging.
operationId: readPersonsCollectingItems
tags:
- Items
parameters:
- $ref: 'commons.yaml#/parameters/pageSize'
- $ref: 'commons.yaml#/parameters/pageNumber'
- $ref: '#/parameters/filterItemTypes'
responses:
'200':
description: A collected items list
schema:
$ref: '#/definitions/PagedCollectingItems'
headers:
$ref: commons.yaml#/defaultHeaders
examples:
application/json:
{
"totalItems": 10,
"totalPage": 4,
"pageSize": 3,
"currentPage": 2,
"items":
[
{
"itemType": "Vinyl",
"maxPrice": 20,
"imageId": "98096838-04eb-4bac-b32e-cd5b7196de71",
"albumName": "Captain Future Original Soundtrack",
"artist": "Yuji Ohno"
},
{
"itemType": "VHS",
"maxPrice": 10,
"imageId": "b74469bc-e6a1-4a90-858a-88ef94079356",
"movieTitle": "Star Crash",
"director": "Luigi Cozzi"
},
{
"itemType": "AudioCassette",
"maxPrice": 10,
"imageId": "b74469bc-e6a1-4a90-858a-88ef94079356",
"albumName": "Star Wars",
"artist": "John Williams"
}
]
}
'404':
$ref: '#/responses/PersonDoesNotExistResponse'
'500':
$ref: 'commons.yaml#/responses/Standard500ErrorResponse'
default:
$ref: 'commons.yaml#/responses/TotallyUnexpectedResponse'
definitions:
Person:
title: Human
description: A person which can be the user itself or one of his friend
required:
- username
properties:
firstName:
description: first name
type: string
example: John
lastName:
description: last name
type: string
example: Doe
username:
description: Username used to connect to the service
type: string
pattern: '[a-z0-9]{8,64}'
minLength: 8
maxLength: 64
example: john1doe6
dateOfBirth:
description: Date of birth
type: string
format: date
example: 1978-06-21
lastTimeOnline:
description: The last time this person was connected to the service as a
type: string
format: date-time
readOnly: true
example: 2016-06-10T12:36:58.014Z
avatarBase64PNG:
description: An avatar PNG image as a base64 encoded string ready to use as an src in img html tag
type: string
format: byte
default: 
spokenLanguages:
$ref: '#/definitions/SpokenLanguages'
SpokenLanguages:
title: Languages
description: A hashmap of spoken languages
additionalProperties:
description: An additional spoken language
type: string
properties:
defaultLanguage:
description: Default spoken language
type: string
default: english
example:
defaultLanguage: french
it: italian
fr: french
Persons:
title: Humans
description: A list of users or friends
required:
- items
properties:
items:
description: Array containg the list
type: array
minItems: 10
maxItems: 100
uniqueItems: true
items:
$ref: '#/definitions/Person'
example:
- firstname: Robert
lastname": Doe
username": robdo
dateOfBirth: 1970-01-28
lastTimeOnline: 2016-04-10T14:36:58.014Z
- firstname: Jane
lastname: Doe
username: jdoe123
dateOfBirth: 1980-05-12
lastTimeOnline: 2016-05-12T19:23:59.014Z
CollectingItem:
discriminator: itemType
required:
- itemType
properties:
itemType:
description: |
An item can be of different type:
type | definition
-----|-----------
Vinyl| #/definitions/Vinyl
VHS | #/definitions/VHS
AudioCassette | #/definitions/AudioCassette
type: string
enum:
- AudioCassette
- Vinyl
- VHS
imageId:
type: string
maxPrice:
type: number
format: double
minimum: 0
maximum: 10000
exclusiveMinimum: true
exclusiveMaximum: false
Vinyl:
allOf:
- $ref: '#/definitions/CollectingItem'
- required:
- albumName
- artist
properties:
albumName:
type: string
artist:
type: string
VHS:
allOf:
- $ref: '#/definitions/CollectingItem'
- required:
- movieTitle
properties:
movieTitle:
type: string
director:
type: string
AudioCassette:
allOf:
- $ref: '#/definitions/CollectingItem'
- required:
- albumName
- artist
properties:
albumName:
type: string
artist:
type: string
PagedPersons:
allOf:
- $ref: '#/definitions/Persons'
- $ref: 'commons.yaml#/definitions/Paging'
PagedCollectingItems:
allOf:
- properties:
items:
type: array
minItems: 10
maxItems: 100
uniqueItems: true
items:
$ref: '#/definitions/CollectingItem'
- $ref: 'commons.yaml#/definitions/Paging'
responses:
PersonDoesNotExistResponse:
description: Person does not exist.
headers:
$ref: commons.yaml#/defaultHeaders
parameters:
username:
name: username
in: path
required: true
description: The person's username
type: string
includeNonVerifiedUsers:
name: includeNonVerifiedUsers
in: query
description: Result will not include non verified user by default if this parameter is not provided
type: boolean
default: false
allowEmptyValue: true
sortPersons:
name: sort
in: query
description: Result will be sorted by lastTimeOnline descending and username ascending by default if this parameter is not provided
type: array
uniqueItems: true
minItems: 1
maxItems: 3
collectionFormat: pipes
items:
type: string
pattern: '[-+](username|lastTimeOnline|firstname|lastname)'
default:
- -lastTimeOnline
- +username
filterItemTypes:
name: itemType
in: query
description: Filter collected items on their type
type: array
collectionFormat: multi
uniqueItems: true
items:
type: string
enum:
- AudioCassette
- Vinyl
- VHS
full main file
Here’s the full main file where we reference the 4 sub-files:
swagger: '2.0'
info:
version: 1.1.0
title: Simple API
description: |
A simple API to learn how to write OpenAPI Specification.
This file uses almost every single aspect of the [Open API Specification](https://openapis.org/).
This API will use JSON.
JSON looks like this:
```JSON
{
"key": "value",
"anotherKey": "anotherValue"
}
```
termsOfService: http://simple.api/terms-of-service
contact:
name: John Doe
url: http://simple.api/contact
email: [email protected]
license:
name: Apache-2.0
url: http://www.apache.org/licenses/LICENSE-2.0
externalDocs:
description: |
**Complete** documentation describing how to use this API
url: http://doc.simple.api/
tags:
- name: Persons
description: Everything you need to handle `users` and `friends`
externalDocs:
description: People category documentation
url: http://doc.simple.api/people
- name: Items
description: Everything you need to handle items collected by users
externalDocs:
description: Items category documentation
url: http://doc.simple.api/items
- name: Media
description: Everything you need to handle images
externalDocs:
description: Media category documentation
url: http://doc.simple.api/media
- name: JSLess
description: Specific operations for JS less consumers
externalDocs:
description: JS Less Consumers documentation
url: http://doc.simple.api/jsless
schemes:
- https
host: simple.api
basePath: /openapi101
consumes:
$ref: commons.yaml#/defaultMediatypes
produces:
$ref: commons.yaml#/defaultMediatypes
securityDefinitions:
$ref: commons.yaml#/securityDefinitions
security:
$ref: commons.yaml#/defaultSecurity
paths:
/persons:
$ref: 'persons.yaml#/persons'
/js-less-consumer-persons:
$ref: 'legacy.yaml#/js-less-consumer-persons'
'/persons/{username}':
$ref: 'persons.yaml#/persons-username'
'/persons/{username}/friends':
$ref: 'persons.yaml#/persons-username-friends'
'/persons/{username}/collecting-items':
$ref: 'persons.yaml#/persons-username-collecting-items'
/images:
$ref: 'images.yaml#/images'
/images/{imageId}:
$ref: 'images.yaml#/images-imageId'
Valid sub-files
If we put the last persons.yaml file content in the editor, it ends with these errors:
Splitting a huge Open API Specification file to keep it maintainable is a great idea but if you cannot validate sub-files against the specification, you gain almost nothing.
Making commons.yaml valid
If we put the commons.yaml file created earlier in the editor we get these errors:
- Missing required property: swagger
- Missing required property: info
- Missing required property: paths
- Additional properties not allowed: defaultHeaders,defaultMediatypes,defaultSecurity
Let’s see how we can fix these errors.
Missing swagger property
We just need to add this line on file’s top:
Missing info property
We add an short info
section after swagger
:
Missing paths property
As we do not define any path in this file we just need to add an empty paths
section:
Property defaultSecurity not allowed
As the defaultSecurity
correspond to the OpenAPI security
section we just need to rename it:
Invalid commons.yaml
file:
Valid commons.yaml
file:
Property defaultMediatypes not allowed
defaultMediaType
is not a valid OpenAPI property:
We will use produces
and consumes
to define the default media types. But as we want to define a single set of media type we will use this trick:
In the future if we want to define different sets of media types for produces and consumes, we’ll be ready.
Property defaultHeaders not allowed
defaultHeaders
is not a valid OpenAPI property:
defaultHeaders:
X-Rate-Limit-Remaining:
description: How many calls consumer can do
type: integer
X-Rate-Limit-Reset:
description: When rate limit will be reset
type: string
format: date-time
We will use a dummy response definition DefaultHeaders
to declare these default headers:
responses:
DefaultHeaders:
description: A dummy response to define default header
headers:
X-Rate-Limit-Remaining:
description: How many calls consumer can do
type: integer
X-Rate-Limit-Reset:
description: When rate limit will be reset
type: string
format: date-time
Of course we need to update common responses to use these headers:
Standard500ErrorResponse:
description: An unexpected error occured.
headers:
$ref: '#/responses/DefaultHeaders/headers'
schema:
$ref: '#/definitions/Error'
TotallyUnexpectedResponse:
description: A totally unexpected response
headers:
$ref: '#/responses/DefaultHeaders/headers'
Making persons.yaml valid
Just like with commons.yaml
, if we put persons.yaml
content in the editor we get some errors:
- Missing required property: swagger
- Missing required property: info
- Reference could not be resolved: commons.yaml#/defaultHeaders
- Missing required property: paths
- Additional properties not allowed: persons-username-collecting-items,persons-username-friends,persons-username,persons
Missing swagger and info properties
We just add swagger
and info
like for commons.yaml
:
swagger: "2.0"
info:
version: 1.0.0
title: Persons Sub-API
description: Persons operations, definitions, parameters and responses
securityDefinitions:
$ref: commons.yaml#/securityDefinitions
commons.yaml#/defaultHeaders reference could not be resolved
As the commons.yaml
structure has been modified concerning default headers we need to replace these references:
by this one:
Missing paths and additional properties not allowed
We have defined custom properties for each path relative to persons operations, therefore there is no paths
section and our custom properties (like persons
) are not allowed by the OpenAPI specification.
persons:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
operationId: searchUsers
We need to add a paths
section and put all our path in it using the correct path value as key:
paths:
'/persons':
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
operationId: searchUsers
New errors: Security definition could not be resolved
Once this is done, 2 new errors appear:
- Security definition could not be resolved: OauthSecurity
- Security definition could not be resolved: LegacySecurity
Now that we have a valid structure, the paths have been parsed and the parser detected missing security definitions. To solve this error, we add these definitions by referencing the commons.yaml
file:
images.yaml and legacy.yaml
For images.yaml
and legacy.yaml
we do exactly the same things as we’ve done with persons.yaml
.
Here the images.yaml
file modified (partial view):
swagger: "2.0"
info:
version: 1.0.0
title: Images Sub-API
description: images operations
securityDefinitions:
$ref: commons.yaml#/securityDefinitions
paths:
/images:
Here the legacy.yaml
file modified (partial view):
swagger: "2.0"
info:
version: 1.0.0
title: Legacy Sub-API
description: Legacy operations
securityDefinitions:
$ref: commons.yaml#/securityDefinitions
paths:
/js-less-consumer-persons:
Updating the main file
Now that we have modified all sub-files, if we try to edit the main file we get these errors
- Reference could not be resolved: commons.yaml#/defaultMediatypes
- Reference could not be resolved: commons.yaml#/defaultSecurity
- Reference could not be resolved: persons.yaml#/persons
- Reference could not be resolved: images.yaml#/images-imageId
- Reference could not be resolved: persons.yaml#/persons-username
- Reference could not be resolved: persons.yaml#/persons-username-friends
- Reference could not be resolved: persons.yaml#/persons-username-collecting-items
- Reference could not be resolved: images.yaml#/images
- Reference could not be resolved: legacy.yaml#/js-less-consumer-persons
These errors are of 2 types:
- those due to the
commons.yaml
structure modification - and those due to the
persons.yaml
,images.yaml
andlegacy.yaml
paths structure modifications
Modifiying commons.yaml references
Before:
consumes:
$ref: commons.yaml#/defaultMediatypes
produces:
$ref: commons.yaml#/defaultMediatypes
securityDefinitions:
$ref: commons.yaml#/securityDefinitions
security:
$ref: commons.yaml#/defaultSecurity
After:
consumes:
$ref: 'commons.yaml#/consumes'
produces:
$ref: 'commons.yaml#/produces'
securityDefinitions:
$ref: 'commons.yaml#/securityDefinitions'
security:
$ref: 'commons.yaml#/security'
Modifying paths references
This is the trickyest part of this tutorial. To reference the /persons
path in persons.yaml
we used persons.yaml#/persons
:
But persons.yaml
structure has changed from :
persons:
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
operationId: searchUsers
to:
paths:
'/persons':
parameters:
- $ref: 'commons.yaml#/parameters/userAgent'
get:
summary: Gets some persons
description: Returns a list containing all persons. The list supports paging.
operationId: searchUsers
The new reference should be something like persons.yaml#/paths//persons
but if we try it, the editor shows an error Reference could not be resolved: persons.yaml#/paths//persons. The /
in /persons
needs to be escaped, how can we do that?
Evaluation of each reference token begins by decoding any escaped character sequence. This is performed by first transforming any occurrence of the sequence ‘~1’ to ‘/’ RFC6901
Before modification:
paths:
/persons:
$ref: 'persons.yaml#/persons'
/js-less-consumer-persons:
$ref: 'legacy.yaml#/js-less-consumer-persons'
'/persons/{username}':
$ref: 'persons.yaml#/persons-username'
'/persons/{username}/friends':
$ref: 'persons.yaml#/persons-username-friends'
'/persons/{username}/collecting-items':
$ref: 'persons.yaml#/persons-username-collecting-items'
/images:
$ref: 'images.yaml#/images'
/images/{imageId}:
$ref: 'images.yaml#/images-imageId'
After modification (all /
in reference name have been replaced by ~1
):
paths:
/persons:
$ref: 'persons.yaml#/paths/~1persons'
/js-less-consumer-persons:
$ref: 'legacy.yaml#/paths/~1js-less-consumer-persons'
'/persons/{username}':
$ref: 'persons.yaml#/paths/~1persons~1{username}'
'/persons/{username}/friends':
$ref: 'persons.yaml#/paths/~1persons~1{username}~1friends'
'/persons/{username}/collecting-items':
$ref: 'persons.yaml#/paths/~1persons~1{username}~1collecting-items'
/images:
$ref: 'images.yaml#/paths/~1images'
/images/{imageId}:
$ref: 'images.yaml#/paths/~1images~1{imageId}'
And now the main file and all its sub-files are considered valid by the editor.
Conclusion
You are now ready to split any OpenAPI specification file into valid sub-files in any possible ways. In this 8th part you’ve learned how to use JSON pointers in almost any places in an OpenAPI specification to references items from other files and how to create valid sub-files containing partial information. In the next final part (at last) we will learn how to extend the OpenAPI specification.