Skip to main content

OpenAPI Overview

OpenAPI is a standard for representing API specifications. It is useful whether your service will be consumed by other services or front end clients such as mobile devices. Psychic and Dream work together to automatically generate OpenAPI 3.0.2 specs for your web application with as little additional work from you as possible. They do this by leveraging:

  1. the routes already defined in api/src/conf/routes.ts
  2. the serializers you already use to turn models into JSON
  3. an @OpenAPI decorator in controllers

The resource generator automatically adds @OpenAPI decorator declarations corresponding to each endpoint, and since serializer attributes are typed back to the database schema, these types will stay in sync as you add migrations. (For example, if you add a migration to change a column from not allowing null to allowing null, the OpenAPI spec will change to include nullable: true for the corresponding property.)

Serializers and the @OpenAPI decorator also support hand-written specs to provide added flexibility when needed. As with everything in Dream and Psychic, these hand-written specs are strongly typed.

OpenAPI validation is enabled on all requests (when a request doesn't match, it returns 400) and on all responses when running specs (to ensure that any hand-written OpenAPI specifications match the data being returned).

To generate the OpenAPI spec, simply run:

yarn psy sync:openapi

Psychic provides built-in support for generating openapi schema to define your response payloads. Using the definitions you provide, Psychic will regenerate an openapi.json file at the root of your project whenever a sync occurs. Psychic will also use this to generate type files for your client app, allowing you to easily sync up your api mechanisms to the current values provided by the backend.

Automatic response generation

export default class PostsController extends PsychicController {
@OpenAPI(Post, {
many: true,
status: 200,
serializerKey: 'summary',
})
public async index() {
this.ok(await this.currentUser.associationQuery('posts').all())
}
}

If the first argument to the OpenAPI decorator is a callback function, then the return value of that function will dictate the success response payload. In the above case, because the many: true flag has been passed, the openapi response will yield an array type for the response content:

"paths": {
"/posts": {
"parameters": [],
"get": {
"tags": [],
"summary": "",
"responses": {
"200": {
"content": {
"application/json": {
"schema": {
"type": "array",
"items": {
"$ref": "#/components/schemas/PostSummary"
}
}
}
},
"description": "index"
}
}
}
},
}

Implicit serializer scanning

In the above example, a reference to the #/components/schemas/PostSummary component is made. This component is also generated in the components section of the openapi.json, leveraging the @Attribute decorator calls to construct an image of the serializer when building component schemas.

"components": {
"schemas": {
"PostSummary": {
"type": "object",
"required": [
"id",
"body",
],
"properties": {
"id": {
"type": "string"
},
"body": {
"type": "string"
},
}
},
}
}

Dream supports OpenAPI notation within the @Attribute decorator, allowing us to build openapi specs for each attribute:

class PostSerializer extends DreamSerializer {
@Attribute({
type: 'object',
properties: {
label: 'string',
value: {
type: 'string',
nullable: true,
},
},
})
public body() {
return {
label: 'Body',
value: this.data.body,
}
}
}

Extra response payloads

In addition to the default response shapes described, you can pass custom response objects to the responses field, enabling you to handle custom response codes.

export default class PostsController extends PsychicController {
@OpenAPI(Post, {
many: true,
status: 200,
serializerKey: 'summary',
responses: {
400: {
type: 'object',
properties: {
errors: 'string[]',
},
},
},
})
public async index() {
this.ok(await this.currentUser.associationQuery('posts').all())
}
}

Parameters

To populate the parameters field in your OpenAPI definition for this route, the headers, body, pathParams, and query options are passed:

export default class PostsController extends PsychicController {
@OpenAPI(Post, {
body: {
type: 'object',
properties: {
email: 'string',
},
},
query: [{ name: 'search', required: false }],
headers: [{ name: 'Authorization', required: true }],
pathParams: [{ name: 'id', required: true }],
})
public async create() {
this.ok(await this.currentUser.associationQuery('posts').all())
}
}

Tags

When openapi routes are read, they are often grouped by the fields provided in their tags array. The tags option enables you to populate this field.

export default class PostsController extends PsychicController {
@OpenAPI(Post, {
many: true,
status: 200,
serializerKey: 'summary',
tags: ['posts'],
})
public async index() {
this.ok(await this.currentUser.associationQuery('posts').all())
}
}