Skip to content

Commit f26ba4e

Browse files
committed
Added nbf support
1 parent ab76ec5 commit f26ba4e

File tree

4 files changed

+71
-1
lines changed

4 files changed

+71
-1
lines changed

README.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ encoded private key for RSA and ECDSA.
2626

2727
* `algorithm` (default: `HS256`)
2828
* `expiresInMinutes` or `expiresInSeconds`
29+
* `notBeforeMinutes` or `notBeforeSeconds`
2930
* `audience`
3031
* `subject`
3132
* `issuer`
@@ -37,7 +38,7 @@ encoded private key for RSA and ECDSA.
3738
If `payload` is not a buffer or a string, it will be coerced into a string
3839
using `JSON.stringify`.
3940

40-
If any `expiresInMinutes`, `audience`, `subject`, `issuer`, `jwtid`, `subject` are not provided, there is no default. The jwt generated won't include those properties in the payload.
41+
If any `expiresInMinutes`, `notBeforeMinutes`, `audience`, `subject`, `issuer`, `jwtid`, `subject` are not provided, there is no default. The jwt generated won't include those properties in the payload.
4142

4243
Additional headers can be provided via the `headers` object.
4344

@@ -60,6 +61,7 @@ var token = jwt.sign({ foo: 'bar' }, cert, { algorithm: 'RS256'});
6061
`options`:
6162

6263
* `ignoreExpiration`
64+
* `ignoreNotBefore`
6365
* `audience`
6466
* `issuer`
6567
* `jwtid`
@@ -81,6 +83,7 @@ encoded public key for RSA and ECDSA.
8183
* `audience`: if you want to check audience (`aud`), provide a value here
8284
* `issuer`: if you want to check issuer (`iss`), provide a value here
8385
* `ignoreExpiration`: if `true` do not validate the expiration of the token.
86+
* `ignoreNotBefore`: if `true` do not validate the not before of the token.
8487
* `jwtid`: if you want to check JWT ID (`jti`), provide a value here
8588
* `subject`: if you want to check subject (`sub`), provide a value here
8689

index.js

+17
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
var jws = require('jws');
22

33
var JsonWebTokenError = module.exports.JsonWebTokenError = require('./lib/JsonWebTokenError');
4+
var NotBeforeError = module.exports.NotBeforeError = require('./lib/NotBeforeError');
45
var TokenExpiredError = module.exports.TokenExpiredError = require('./lib/TokenExpiredError');
56

67
module.exports.decode = function (jwt, options) {
@@ -53,6 +54,14 @@ module.exports.sign = function(payload, secretOrPrivateKey, options) {
5354
if (!options.noTimestamp) {
5455
payload.iat = payload.iat || timestamp;
5556
}
57+
58+
var notBeforeSeconds = options.notBeforeMinutes ?
59+
options.notBeforeMinutes * 60 :
60+
options.notBeforeSeconds;
61+
62+
if (notBeforeSeconds) {
63+
payload.nbf = timestamp + notBeforeSeconds;
64+
}
5665

5766
var expiresInSeconds = options.expiresInMinutes ?
5867
options.expiresInMinutes * 60 :
@@ -167,6 +176,14 @@ module.exports.verify = function(jwtString, secretOrPublicKey, options, callback
167176
} catch(err) {
168177
return done(err);
169178
}
179+
180+
if (typeof payload.nbf !== 'undefined' && !options.ignoreNotBefore) {
181+
if (typeof payload.nbf !== 'number') {
182+
return done(new JsonWebTokenError('invalid nbf value'));
183+
}
184+
if (payload.nbf >= Math.floor(Date.now() / 1000))
185+
return done(new NotBeforeError('jwt not active', new Date(payload.nbf * 1000)));
186+
}
170187

171188
if (typeof payload.exp !== 'undefined' && !options.ignoreExpiration) {
172189
if (typeof payload.exp !== 'number') {

lib/NotBeforeError.js

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
var JsonWebTokenError = require('./JsonWebTokenError');
2+
3+
var NotBeforeError = function (message, date) {
4+
JsonWebTokenError.call(this, message);
5+
this.name = 'NotBeforeError';
6+
this.date = date;
7+
};
8+
9+
NotBeforeError.prototype = Object.create(JsonWebTokenError.prototype);
10+
11+
NotBeforeError.prototype.constructor = NotBeforeError;
12+
13+
module.exports = NotBeforeError;

test/jwt.rs.tests.js

+37
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,43 @@ describe('RS256', function() {
8888
});
8989
});
9090

91+
describe('when signing a token with not before', function() {
92+
var token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBeforeMinutes: -10 });
93+
94+
it('should be valid expiration', function(done) {
95+
jwt.verify(token, pub, function(err, decoded) {
96+
assert.isNotNull(decoded);
97+
assert.isNull(err);
98+
done();
99+
});
100+
});
101+
102+
it('should be invalid', function(done) {
103+
// not active token
104+
token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBeforeMinutes: 10 });
105+
106+
jwt.verify(token, pub, function(err, decoded) {
107+
assert.isUndefined(decoded);
108+
assert.isNotNull(err);
109+
assert.equal(err.name, 'NotBeforeError');
110+
assert.instanceOf(err.date, Date);
111+
assert.instanceOf(err, jwt.NotBeforeError);
112+
done();
113+
});
114+
});
115+
116+
it('should NOT be invalid', function(done) {
117+
// not active token
118+
token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBeforeMinutes: 10 });
119+
120+
jwt.verify(token, pub, { ignoreNotBefore: true }, function(err, decoded) {
121+
assert.ok(decoded.foo);
122+
assert.equal('bar', decoded.foo);
123+
done();
124+
});
125+
});
126+
});
127+
91128
describe('when signing a token with audience', function() {
92129
var token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', audience: 'urn:foo' });
93130

0 commit comments

Comments
 (0)