Skip to main content

resources

Psychic provides resourceful routing to encourage devs to think about their app in terms of resources. While the needs of an application will by highly diverse from app to app, they can all be boiled down to a collection of resources which will ultimately fill their databases and represent their user's state within the context of their application.

It is fundamentally essential to think this way, since you are already being forced to do it when you separate out your application's needs into a collection of tables. The idea here is to extend these resourceful representations of your data up into the routing layer as well.

For example, if you are making an application that allows users to sign up and create a blog, you may need a users table, a blogs table, a posts table, and a comments table. It would be nice if your application could cleanly reflect some of these resources:

import { PsychicRouter } from 'psychic'

export default (r: PsychicRouter) => {
r.resources('blogs', (r) => {
r.resources('posts')
})
}

GET     /blogs                          Blogs#index
POST    /blogs                          Blogs#create
GET     /blogs/:id                      Blogs#show
PUT     /blogs/:id                      Blogs#update
PATCH   /blogs/:id                      Blogs#update
DELETE  /blogs/:id                      Blogs#destroy
GET     /blogs/:id/posts                Blogs/Posts#index
POST    /blogs/:id/posts                Blogs/Posts#create
GET     /blogs/:blogId/posts/:id        Blogs/Posts#show
PUT     /blogs/:blogId/posts/:id        Blogs/Posts#update
PATCH   /blogs/:blogId/posts/:id        Blogs/Posts#update
DELETE  /blogs/:blogId/posts/:id        Blogs/Posts#destroy

You can use resourceful routes, but restrict the routes to only a collection. This is dead-useful, since often you are building an application which does express some resourceful routing, but does not need certain routes. For example, you may have no need for the destroy or update methods. Below is an example of resourceful route limiting:

// conf/routes.ts

import { PsychicRouter } from 'psychic'

export default (r: PsychicRouter) => {
r.resources('users', { only: ['create', 'index'] })
// same as:
r.resources('users', { except: ['show', 'update', 'destroy'] })
}

GET     /users  Users#index
POST    /users  Users#create

Additionally, use the resource method to do the same, but with a singularized route endpoint. This is generally done when you do not need to pass an id to identify the resource, such as when the resource will always belong to the authenticated user.

// conf/routes.ts

import { PsychicRouter } from 'psychic'

export default (r: PsychicRouter) => {
r.resource('user')
}

GET    /user  Users#show
POST   /user  Users#create
PUT    /user  Users#update
PATCH  /user  Users#update
DELETE /user  Users#destroy

Both the resource and resources method enable nesting, enabling you to express child routes. By default, all child routes will be treated as members of the resource (meaning, they need to be reached by id).

r.resources('users', { only: ['show'] }, (r) => {
r.get('edit')
})

GET     /users/:id       Users#show
GET     /users/:id/edit  Users#edit

If you do not wish for your nested endpoint to contain the id as part of its signature, you can use the collection method, like so:


GET     /users/:id   Users#show
GET     /users/edit  Users#edit
r.resources('users', { only: ['show'] }, (r) => {
r.collection((r) => {
r.get('edit')
})
})