Skip to content

Commit 2eb602b

Browse files
committed
🤘
0 parents  commit 2eb602b

File tree

5 files changed

+259
-0
lines changed

5 files changed

+259
-0
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
node_modules/
2+
.vscode/

Readme.md

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
# fastify-bearer-auth
2+
3+
*fastify-bearer-auth* provides a simple request hook for the [Fastify][fastify]
4+
web framework.
5+
6+
[fastify]: https://github.com/fastify/fastify
7+
8+
## Example
9+
10+
```js
11+
'use strict'
12+
13+
const fastify = require('fastify')
14+
const bearerAuthPlugin = require('fastify-bearer-auth')
15+
const keys = new Set(['a-super-secret-key', 'another-super-secret-key'])
16+
17+
fastify.addHook('preHandler', bearerAuthPlugin({keys}))
18+
fastify.get('/foo', (req, reply) => {
19+
reply({authenticated: true})
20+
})
21+
22+
fastify.listen({port: 8000}, (err) => {
23+
if (err) {
24+
console.error(err.message)
25+
process.exit(1)
26+
}
27+
console.log.info('http://127.0.0.1:8000/foo')
28+
})
29+
```
30+
31+
## API
32+
33+
+ `factory(config)`: exported by `require('fastify-bearer-auth')`. The `config`
34+
object must have a `keys` property that is set to an object which has a
35+
`has(key)` method. It may also have method `errorResponse(err)` and property
36+
`contentType`. If set, the `errorResponse(err)` method must synchronously
37+
return the content body to be sent to the client. If the content to be sent
38+
is anything other than `application/json`, then the `contentType` property
39+
must be set. The default config object is:
40+
41+
```js
42+
{
43+
keys: new Set(),
44+
contentType: undefined,
45+
errorResponse: (err) => {
46+
return {error: err.message}
47+
}
48+
}
49+
```
50+
51+
+ `bearerAuthHook(req, reply, next)`: a standard *Fastify*
52+
[preHandler hook][prehook] which will inspect the request's headers
53+
for an `authorization` header in the format `bearer key`. The `key` will be
54+
matched against the configured `keys` object via the `has(key)` method. If
55+
the `authorization` header is missing, malformed, or the `key` does not
56+
validate then a 401 response will be sent with a `{error: message}` body;
57+
no further request processing will be performed.
58+
59+
[prehook]: https://github.com/fastify/fastify/blob/master/docs/Hooks.md
60+
61+
## License
62+
63+
[MIT License](http://jsumners.mit-license.org/)

package.json

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
{
2+
"name": "fastify-bearer-auth",
3+
"version": "1.0.0",
4+
"description": "An authentication plugin for Fastify",
5+
"main": "plugin.js",
6+
"scripts": {
7+
"test": "tap 'test/**/*.js'",
8+
"lint": "standard"
9+
},
10+
"precommit": [
11+
"lint",
12+
"test"
13+
],
14+
"repository": {
15+
"type": "git",
16+
"url": "git+ssh://[email protected]/jsumners/fastify-bearer-auth.git"
17+
},
18+
"keywords": [
19+
"fastify",
20+
"authentication"
21+
],
22+
"author": "James Sumners <[email protected]>",
23+
"license": "MIT",
24+
"bugs": {
25+
"url": "https://github.com/jsumners/fastify-bearer-auth/issues"
26+
},
27+
"homepage": "https://github.com/jsumners/fastify-bearer-auth#readme",
28+
"devDependencies": {
29+
"pre-commit": "^1.2.2",
30+
"standard": "^10.0.1",
31+
"tap": "^10.3.2"
32+
}
33+
}

plugin.js

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
'use strict'
2+
3+
module.exports = function (options) {
4+
const defaultOptions = {
5+
keys: new Set(),
6+
errorResponse (err) {
7+
return {error: err.message}
8+
},
9+
contentType: undefined
10+
}
11+
const _options = Object.assign(defaultOptions, options || {})
12+
const keys = _options.keys
13+
const errorResponse = _options.errorResponse
14+
const contentType = _options.contentType
15+
16+
function bearerAuthHook (fastifyReq, fastifyRes, next) {
17+
const header = fastifyReq.req.headers['authorization']
18+
if (!header) {
19+
const noHeaderError = Error('missing authorization header')
20+
fastifyReq.log.error('unauthorized: %s', noHeaderError.message)
21+
if (contentType) fastifyRes.header('content-type', contentType)
22+
fastifyRes.code(401).send(errorResponse(noHeaderError))
23+
return
24+
}
25+
26+
const key = header.substring(6).trim()
27+
if (keys.has(key) === false) {
28+
const invalidKeyError = Error('invalid authorization header')
29+
fastifyReq.log.error('invalid authorization header: `%s`', header)
30+
if (contentType) fastifyRes.header('content-type', contentType)
31+
fastifyRes.code(401).send(errorResponse(invalidKeyError))
32+
return
33+
}
34+
35+
next()
36+
}
37+
38+
return bearerAuthHook
39+
}

test/hook.test.js

+122
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,122 @@
1+
'use strict'
2+
3+
const test = require('tap').test
4+
const noop = () => {}
5+
const plugin = require('../')
6+
const key = '123456789012354579814'
7+
const keys = {keys: new Set([key])}
8+
9+
test('hook rejects for missing header', (t) => {
10+
t.plan(2)
11+
12+
const request = {
13+
log: {error: noop},
14+
req: {headers: {}}
15+
}
16+
const response = {
17+
code: () => response,
18+
send: send
19+
}
20+
21+
function send (body) {
22+
t.ok(body.error)
23+
t.match(body.error, /missing authorization header/)
24+
}
25+
26+
const hook = plugin()
27+
hook(request, response)
28+
})
29+
30+
test('hook rejects header without bearer prefix', (t) => {
31+
t.plan(2)
32+
33+
const request = {
34+
log: {error: noop},
35+
req: {
36+
headers: {authorization: key}
37+
}
38+
}
39+
const response = {
40+
code: () => response,
41+
send: send
42+
}
43+
44+
function send (body) {
45+
t.ok(body.error)
46+
t.match(body.error, /invalid authorization header/)
47+
}
48+
49+
const hook = plugin(keys)
50+
hook(request, response)
51+
})
52+
53+
test('hook rejects malformed header', (t) => {
54+
t.plan(2)
55+
56+
const request = {
57+
log: {error: noop},
58+
req: {
59+
headers: {authorization: `bearerr ${key}`}
60+
}
61+
}
62+
const response = {
63+
code: () => response,
64+
send: send
65+
}
66+
67+
function send (body) {
68+
t.ok(body.error)
69+
t.match(body.error, /invalid authorization header/)
70+
}
71+
72+
const hook = plugin(keys)
73+
hook(request, response)
74+
})
75+
76+
test('hook accepts correct header', (t) => {
77+
t.plan(1)
78+
79+
const request = {
80+
log: {error: noop},
81+
req: {
82+
headers: {authorization: `bearer ${key}`}
83+
}
84+
}
85+
const response = {
86+
code: () => response,
87+
send: send
88+
}
89+
90+
function send (body) {
91+
t.fail('should not happen')
92+
}
93+
94+
const hook = plugin(keys)
95+
hook(request, response, () => {
96+
t.pass()
97+
})
98+
})
99+
100+
test('hook accepts correct header with extra padding', (t) => {
101+
t.plan(1)
102+
103+
const request = {
104+
log: {error: noop},
105+
req: {
106+
headers: {authorization: `bearer ${key} `}
107+
}
108+
}
109+
const response = {
110+
code: () => response,
111+
send: send
112+
}
113+
114+
function send (body) {
115+
t.fail('should not happen')
116+
}
117+
118+
const hook = plugin(keys)
119+
hook(request, response, () => {
120+
t.pass()
121+
})
122+
})

0 commit comments

Comments
 (0)