Skip to content
This repository was archived by the owner on Jul 5, 2022. It is now read-only.

Commit 8197bf7

Browse files
committed
feat: add links to resource objects and relationships
1 parent bce7b8e commit 8197bf7

File tree

3 files changed

+140
-14
lines changed

3 files changed

+140
-14
lines changed

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
"dependencies": {
1919
"bluebird": "^3.1.1",
2020
"lodash": "^3.10.1",
21+
"pluralize": "^3.0.0",
2122
"sequelize": "^3.17.1"
2223
},
2324
"devDependencies": {

src/jsonapi.js

+78-4
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,85 @@
1+
import _ from 'lodash'
2+
import pluralize from 'pluralize'
3+
4+
function documentLinks(req, document) {
5+
const {id, type} = document
6+
return {
7+
self: req.protocol + '://' + req.get('host') + '/' + pluralize(type) + '/' + id
8+
}
9+
}
10+
11+
function relationshipLinks(req, document, relationship) {
12+
const {id, type} = document
13+
const baseUrl = req.protocol + '://' + req.get('host') + '/' + pluralize(type) + '/' + id
14+
return {
15+
self: baseUrl + '/relationships/' + relationship,
16+
related: baseUrl + '/' + relationship
17+
}
18+
}
19+
20+
function links(req, res) {
21+
const body = res.body
22+
23+
// resource objects
24+
if (_.isArray(body.data) && body.data.length > 0) {
25+
body.data.forEach(document => {
26+
document.links = documentLinks(req, document)
27+
if (document.relationships) {
28+
_.forEach(document.relationships, (val, key) => {
29+
document.relationships[key].links = relationshipLinks(req, document, key)
30+
})
31+
}
32+
})
33+
} else if (_.isObject(body.data)) {
34+
body.data.links = documentLinks(req, body.data)
35+
if (body.data.relationships) {
36+
_.forEach(body.data.relationships, (val, key) => {
37+
body.data.relationships[key].links = relationshipLinks(req, body.data, key)
38+
})
39+
}
40+
}
41+
42+
// includes
43+
if (_.isArray(body.included) && body.included.length > 0) {
44+
body.included.forEach(document => {
45+
document.links = documentLinks(req, document)
46+
})
47+
}
48+
49+
// body
50+
if (_.isArray(body.data) && body.data.length > 0) {
51+
const type = body.data[0].type
52+
const baseUrl = req.protocol + '://' + req.get('host') + '/' + pluralize(type)
53+
return {
54+
self: baseUrl
55+
}
56+
} else if (_.isObject(res.body.data)) {
57+
const {id, type} = res.body.data
58+
const baseUrl = req.protocol + '://' + req.get('host') + '/' + pluralize(type) + '/' + id
59+
return {
60+
self: baseUrl
61+
}
62+
}
63+
}
64+
165
export default function() {
266
return function(req, res) {
3-
if (!res.body) {
4-
return res
5-
.status(204)
6-
.end()
67+
if (_.isArray(res.body.data)) {
68+
res.body.data.forEach(document => documentLinks(req, document))
69+
} else if (_.isObject(res.body.data)) {
70+
documentLinks(req, res.body.data)
771
}
872

73+
if (_.isArray(res.body.included)) {
74+
res.body.included.forEach(document => documentLinks(req, document))
75+
}
76+
77+
if (res.meta) {
78+
res.body.meta = res.meta
79+
}
80+
81+
res.body.links = links(req, res)
82+
983
res.json(res.body)
1084
}
1185
}

test/test.js

+61-10
Original file line numberDiff line numberDiff line change
@@ -61,24 +61,45 @@ describe('netiam', () => {
6161
]
6262
})
6363
})
64-
.then(users => {
65-
/*console.log(
66-
util.inspect(
67-
users.map(user => user.toJSON()),
68-
{depth: null}
69-
)
70-
)*/
71-
})
7264
.then(() => done())
7365
.catch(done)
7466
})
7567

7668
it('should fetch user', done => {
7769
request(app)
78-
.get(`/users/${user.id}`)
70+
.get(`/users/${user.id}?include=projects`)
7971
.set('Accept', 'application/json')
8072
.expect(200)
8173
.expect('Content-Type', /json/)
74+
.expect(res => {
75+
const json = res.body
76+
77+
json.should.be.an.Object()
78+
json.should.have.properties(['data', 'included', 'links'])
79+
json.data.should.be.Object()
80+
json.data.should.have.properties([
81+
'id',
82+
'type',
83+
'attributes',
84+
'relationships',
85+
'links'
86+
])
87+
json.data.relationships.should.be.Object()
88+
json.data.relationships.should.have.properties([
89+
'projects'
90+
])
91+
92+
json.included.should.be.Array()
93+
json.included.should.have.length(1)
94+
json.included[0].should.have.properties([
95+
'id',
96+
'type',
97+
'attributes',
98+
'links'
99+
])
100+
json.included[0].type.should.eql('project')
101+
json.included[0].links.should.have.properties(['self'])
102+
})
82103
.end(err => {
83104
if (err) {
84105
return done(err)
@@ -89,10 +110,40 @@ describe('netiam', () => {
89110

90111
it('should fetch users', done => {
91112
request(app)
92-
.get('/users')
113+
.get('/users?include=projects')
93114
.set('Accept', 'application/json')
94115
.expect(200)
95116
.expect('Content-Type', /json/)
117+
.expect(res => {
118+
const json = res.body
119+
120+
json.should.be.an.Object()
121+
json.should.have.properties(['data', 'included', 'links'])
122+
json.data.should.be.Array()
123+
json.data.should.have.length(10)
124+
json.data[0].should.have.properties([
125+
'id',
126+
'type',
127+
'attributes',
128+
'relationships',
129+
'links'
130+
])
131+
json.data[0].relationships.should.be.Object()
132+
json.data[0].relationships.should.have.properties([
133+
'projects'
134+
])
135+
136+
json.included.should.be.Array()
137+
json.included.should.have.length(1)
138+
json.included[0].should.have.properties([
139+
'id',
140+
'type',
141+
'attributes',
142+
'links'
143+
])
144+
json.included[0].type.should.eql('project')
145+
json.included[0].links.should.have.properties(['self'])
146+
})
96147
.end(err => {
97148
if (err) {
98149
return done(err)

0 commit comments

Comments
 (0)