Skip to content

Commit 46372e9

Browse files
committed
improvements to nbf and jti claims
1 parent a3c5c23 commit 46372e9

File tree

5 files changed

+38
-24
lines changed

5 files changed

+38
-24
lines changed

README.md

+2-3
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ encoded private key for RSA and ECDSA.
2828

2929
* `algorithm` (default: `HS256`)
3030
* `expiresIn`: expressed in seconds or an string describing a time span [rauchg/ms](https://github.com/rauchg/ms.js). Eg: `60`, `"2 days"`, `"10h"`, `"7d"`
31-
* `notBeforeMinutes` or `notBeforeSeconds`
31+
* `notBefore`: expressed in seconds or an string describing a time span [rauchg/ms](https://github.com/rauchg/ms.js). Eg: `60`, `"2 days"`, `"10h"`, `"7d"`
3232
* `audience`
3333
* `subject`
3434
* `issuer`
@@ -80,8 +80,7 @@ encoded public key for RSA and ECDSA.
8080
* `audience`: if you want to check audience (`aud`), provide a value here
8181
* `issuer`: if you want to check issuer (`iss`), provide a value here
8282
* `ignoreExpiration`: if `true` do not validate the expiration of the token.
83-
* `ignoreNotBefore`...
84-
* `jwtid`: if you want to check JWT ID (`jti`), provide a value here
83+
* `ignoreNotBefore`...
8584
* `subject`: if you want to check subject (`sub`), provide a value here
8685

8786
```js

index.js

+13-18
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
var jws = require('jws');
22
var ms = require('ms');
3+
var timespan = require('./lib/timespan');
34

45
var JWT = module.exports;
56

@@ -57,13 +58,12 @@ JWT.sign = function(payload, secretOrPrivateKey, options, callback) {
5758
if (!options.noTimestamp) {
5859
payload.iat = payload.iat || timestamp;
5960
}
60-
61-
var notBeforeSeconds = options.notBeforeMinutes ?
62-
options.notBeforeMinutes * 60 :
63-
options.notBeforeSeconds;
64-
65-
if (notBeforeSeconds) {
66-
payload.nbf = timestamp + notBeforeSeconds;
61+
62+
if (options.notBefore) {
63+
payload.nbf = timespan(options.notBefore);
64+
if (typeof payload.nbf === 'undefined') {
65+
throw new Error('"notBefore" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60');
66+
}
6767
}
6868

6969
if (options.expiresInSeconds || options.expiresInMinutes) {
@@ -83,15 +83,8 @@ JWT.sign = function(payload, secretOrPrivateKey, options, callback) {
8383

8484
payload.exp = timestamp + expiresInSeconds;
8585
} else if (options.expiresIn) {
86-
if (typeof options.expiresIn === 'string') {
87-
var milliseconds = ms(options.expiresIn);
88-
if (typeof milliseconds === 'undefined') {
89-
throw new Error('bad "expiresIn" format: ' + options.expiresIn);
90-
}
91-
payload.exp = timestamp + milliseconds / 1000;
92-
} else if (typeof options.expiresIn === 'number' ) {
93-
payload.exp = timestamp + options.expiresIn;
94-
} else {
86+
payload.exp = timespan(options.expiresIn);
87+
if (typeof payload.exp === 'undefined') {
9588
throw new Error('"expiresIn" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60');
9689
}
9790
}
@@ -211,13 +204,15 @@ JWT.verify = function(jwtString, secretOrPublicKey, options, callback) {
211204
} catch(err) {
212205
return done(err);
213206
}
214-
207+
215208
if (typeof payload.nbf !== 'undefined' && !options.ignoreNotBefore) {
216209
if (typeof payload.nbf !== 'number') {
217210
return done(new JsonWebTokenError('invalid nbf value'));
218211
}
219-
if (payload.nbf >= Math.floor(Date.now() / 1000))
212+
if (payload.nbf >= Math.floor(Date.now() / 1000)) {
213+
console.log(payload.nbf, '>=', Math.floor(Date.now() / 1000));
220214
return done(new NotBeforeError('jwt not active', new Date(payload.nbf * 1000)));
215+
}
221216
}
222217

223218
if (typeof payload.exp !== 'undefined' && !options.ignoreExpiration) {

lib/timespan.js

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
var ms = require('ms');
2+
3+
module.exports = function (time) {
4+
var timestamp = Math.floor(Date.now() / 1000);
5+
6+
if (typeof time === 'string') {
7+
var milliseconds = ms(time);
8+
if (typeof milliseconds === 'undefined') {
9+
return;
10+
}
11+
return Math.floor(timestamp + milliseconds / 1000);
12+
} else if (typeof time === 'number' ) {
13+
return timestamp + time;
14+
} else {
15+
return;
16+
}
17+
18+
};

test/expires_format.tests.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ describe('expires option', function() {
2727
it('should throw if expires has a bad string format', function () {
2828
expect(function () {
2929
jwt.sign({foo: 123}, '123', { expiresIn: '1 monkey' });
30-
}).to.throw(/bad "expiresIn" format: 1 monkey/);
30+
}).to.throw(/"expiresIn" should be a number of seconds or string representing a timespan/);
3131
});
3232

3333
it('should throw if expires is not an string or number', function () {

test/jwt.rs.tests.js

+4-2
Original file line numberDiff line numberDiff line change
@@ -89,10 +89,12 @@ describe('RS256', function() {
8989
});
9090

9191
describe('when signing a token with not before', function() {
92-
var token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBeforeMinutes: -10 });
92+
var token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBefore: -10 * 3600 });
9393

9494
it('should be valid expiration', function(done) {
9595
jwt.verify(token, pub, function(err, decoded) {
96+
console.log(token);
97+
console.dir(arguments);
9698
assert.isNotNull(decoded);
9799
assert.isNull(err);
98100
done();
@@ -101,7 +103,7 @@ describe('RS256', function() {
101103

102104
it('should be invalid', function(done) {
103105
// not active token
104-
token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBeforeMinutes: 10 });
106+
token = jwt.sign({ foo: 'bar' }, priv, { algorithm: 'RS256', notBefore: '10m' });
105107

106108
jwt.verify(token, pub, function(err, decoded) {
107109
assert.isUndefined(decoded);

0 commit comments

Comments
 (0)