Skip to main content

OpenAPI - Controllers

Controllers provide an entry point for all openapi entries. When Psychic builds openapi files for your application, it starts by scanning all of your controllers and finding all openapi decorators. It then computes the schema for those endpoints, and then spits that schema out into the corresponding json files.

To leverage the OpenAPI decorator in your controllers, simply import it and use it to decorate the methods you wish to expose, like so:

import { OpenAPI } from '@rvoh/psychic'

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())
}
}

By default, Psychic will match this endpoint to the corresponding route entry in your conf/routes.ts file. If it does not find a matching route, it will raise an exception, so be sure to add a matching entry pointing to that method on your controller, i.e.

// conf/routes.ts
export default function routes(r: PsychicRouter) {
r.resources('posts', { only: ['index'] })
// OR
r.get('/posts', PostsController, 'index')
}

Implicit serialization

If a serializable class is provided as the first argument to the @OpenAPI decorator, it will automatically read all related @Attribute decorators on the corresponding serializer, formulating an object shape that it injects into the openapi document.

export default class PostsController extends PsychicController {
// by providing Post here, we are telling OpenAPI to locate the default
// serializer attached to the Post model and serialize its attribute shapes
// into an openapi document, which will become the 200 response shape for
// this endpoint
@OpenAPI(Post, {
status: 200,
})
public async show() {
this.ok(
await this.currentUser
.associationQuery('posts')
.findOrFail(this.castParam('id', 'bigint')),
)
}
}

In addition to providing a model class as the first argument, you can also specify a serializer directly, which takes away the implicit decision making that Psychic does to decide on a serializer for you.

@OpenAPI(PostSerializer, {
status: 200,
})

serializerKey

You can provide a serializerKey option, which will inform the decorator to use a specific serializer attached to your model, like so:

// models/Post.ts
export default class Post extends ApplicationModel {
public get table() {
return 'posts' as const
}

public get serializers(): DreamSerializers<Post> {
return { default: 'PostSerializer', admin: 'Admin/PostSerializer' }
}

// ...
}

// controllers/Admin/PostsController
export default class AdminPostsController extends PsychicController {
// by specifying the "admin" serializerKey, we are telling OpenAPI
// to use the `Admin/PostSerializer`.
@OpenAPI(Post, {
serializerKey: 'admin',
status: 200,
})
public async show() {
this.ok(
await this.currentUser
.associationQuery('posts')
.findOrFail(this.castParam('id', 'bigint')),
)
}
}

many

Use the many option when you are providing a serializable class, and you want the endpoint to render an array of them. This can be done like so:

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

nullable

As expected, the nullable option enables you to specify that the default response can also be null.

export default class AdminPostsController extends PsychicController {
@OpenAPI(Post, {
nullable: true,
status: 200,
})
public async show() {
this.ok(await this.currentUser.associationQuery('posts').first())
}
}

status

By providing a status option to the OpenAPI decorator, you can specify which HTTP status code will be returned if the request succeeds.

export default class PostsController extends PsychicController {
@OpenAPI({
status: 204,
})
public async helloWorld() {
this.noContent()
}
}

requestBody

Using the requestBody option, we can inform OpenAPI what the shape of our incoming request will be, using standard OpenAPI notation.

export default class PostsController extends PsychicController {
@OpenAPI({
status: 204,
requestBody: {
type: 'object',
required: ['searchTerm'],
properties: {
searchTerm: {
type: 'string',
description: 'A search term provided by the user',
},
},
},
})
public async helloWorld() {
this.noContent()
}
}

requestBody

Using the requestBody option, we can inform OpenAPI what the shape of our incoming request will be, using standard OpenAPI notation.

export default class UsersController extends PsychicController {
@OpenAPI({
requestBody: {
type: 'object',
required: ['firstName', 'lastName'],
properties: {
firstName: {
type: 'string',
description: "The user's first name",
},
lastName: {
type: 'string',
description: "The user's first name",
},
},
},
})
public async create() {
await User.create(this.paramsFor(User, { only: ['firstName', 'lastName'] }))
this.noContent()
}
}

headers

Provide the headers option to specify which headers are expected for this endpoint.

export default class PostsController extends PsychicController {
@OpenAPI({
headers: {
Authorization: { description: 'Bearer token', required: true },
},
})
public async helloWorld() {
this.noContent()
}
}

responses

Use the responses option to specify the payload shape for specific response statuses, like so:

export default class PostsController extends PsychicController {
@OpenAPI({
status: 204,
responses: {
400: {
type: 'object',
properties: {
errors: 'string[]',
},
},
},
})
public async helloWorld() {
if (something) {
this.badRequest({ errors: ['error 1', 'error 2'] })
}

this.noContent()
}
}

query

The query option enables you to specify the shape of custom query params for your endpoint. By default, all query params are automatically assumed to be strings, but you can customize their shape if desired, using the schema suboption, as shown below

export default class PostsController extends PsychicController {
@OpenAPI({
query: {
searchTerm: {
required: false,
},
searchTerms: {
required: false,
schema: {
type: 'string[]',
},
},
},
})
public async helloWorld() {
this.noContent()
}
}

defaultResponse

Use the defaultResponse option to specify attributes for the default response, such as description:

export default class PostsController extends PsychicController {
@OpenAPI({
defaultResponse: {
description: 'my description',
},
})
public async helloWorld() {
this.noContent()
}
}

security

Use the security option to provide security entries for openapi. The shape provided here is identical to the shape required by openapi itself, which is an array of objects, where the keys are the names, and the values are arrays of scopes required (or a blank array if there are no scopes).

export default class PostsController extends PsychicController {
@OpenAPI({
security: [
{
bearerToken: ['read', 'write'],
},
{
httpBasicAuth: [],
},
],
})
public async helloWorld() {
this.noContent()
}
}

omitDefaultHeaders

The omitDefaultHeaders option enables you to exclude all default-provided headers (which are configured in conf/app.ts).

export default class PostsController extends PsychicController {
@OpenAPI({
omitDefaultHeaders: true,
})
public async helloWorld() {
this.noContent()
}
}

omitDefaultResponses

Similar to the omitDefaultHeaders option, the omitDefaultResponses option enables you to exclude all default-provided responses (which are also configured in conf/app.ts).

export default class PostsController extends PsychicController {
@OpenAPI({
omitDefaultResponses: true,
})
public async helloWorld() {
this.noContent()
}
}

pathParams

export default class PostsController extends PsychicController {
@OpenAPI({
pathParams: {
id: {
description: 'the id of the post',
},
},
})
public async show() {
this.ok(
await this.currentUser
.associationQuery('posts')
.findOrFail(this.castParam('id', 'bigint')),
)
}
}